bitronix.tm.resource.common.XAPool Maven / Gradle / Ivy
/*
* Copyright (C) 2006-2013 Bitronix Software (http://www.bitronix.be)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package bitronix.tm.resource.common;
import bitronix.tm.BitronixTransaction;
import bitronix.tm.BitronixXid;
import bitronix.tm.TransactionManagerServices;
import bitronix.tm.internal.BitronixRuntimeException;
import bitronix.tm.internal.LogDebugCheck;
import bitronix.tm.internal.XAResourceHolderState;
import bitronix.tm.recovery.IncrementalRecoverer;
import bitronix.tm.recovery.RecoveryException;
import bitronix.tm.resource.common.XAStatefulHolder.State;
import bitronix.tm.utils.MonotonicClock;
import bitronix.tm.utils.Uid;
import javax.transaction.Synchronization;
import java.util.*;
import java.util.concurrent.BlockingDeque;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.logging.Level;
/**
* Generic XA pool. {@link XAStatefulHolder} instances are created by the {@link XAPool} out of a
* {@link XAResourceProducer}. Those objects are then pooled and can be retrieved and/or recycled by the pool
* depending on the running XA transaction's and the {@link XAStatefulHolder}'s states.
*
* @author Ludovic Orban
* @author Brett Wooldridge
*/
public class XAPool, T extends XAStatefulHolder>
implements StateChangeListener
{
private static final java.util.logging.Logger log = java.util.logging.Logger.getLogger(XAPool.class.toString());
/**
* The stateTransitionLock makes sure that transitions of XAStatefulHolders from one state to another
* (movement from one pool to another) are atomic. A ReentrantReadWriteLock allows any number of
* readers to access and iterate the accessiblePool and inaccessiblePool without blocking. Readers
* are blocked only for the instant when a connection is moving between pools. These locks are
* sufficient to protect the collections, which are left intentionally non-concurrent so that failures
* in locking logic will be quickly uncovered.
*/
private final ReentrantReadWriteLock stateTransitionLock = new ReentrantReadWriteLock();
private final BlockingDeque availablePool = new LinkedBlockingDeque<>();
private final Queue accessiblePool = new LinkedList<>();
private final Queue inaccessiblePool = new LinkedList<>();
private final AtomicInteger poolSize = new AtomicInteger();
/**
* This map is used to implement the connection sharing feature of Bitronix.
*/
private final Map> statefulHolderTransactionMap = new ConcurrentHashMap<>();
private final ResourceBean bean;
private final XAResourceProducer xaResourceProducer;
private final Object xaFactory;
private final AtomicBoolean failed = new AtomicBoolean();
private final Object poolGrowthShrinkLock = new Object();
/**
* Constructor XAPool creates a new XAPool instance.
*
* @param xaResourceProducer
* of type XAResourceProducer R, T
* @param bean
* of type ResourceBean
* @param xaFactory
* of type Object
*
* @throws Exception
* when
*/
public XAPool(XAResourceProducer xaResourceProducer, ResourceBean bean, Object xaFactory) throws Exception
{
this.xaResourceProducer = xaResourceProducer;
this.bean = bean;
if (bean.getMaxPoolSize() < 1 || bean.getMinPoolSize() > bean.getMaxPoolSize())
{
throw new IllegalArgumentException("cannot create a pool with min " + bean.getMinPoolSize() + " connection(s) and max " + bean.getMaxPoolSize() + " connection(s)");
}
if (bean.getAcquireIncrement() < 1)
{
throw new IllegalArgumentException("cannot create a pool with a connection acquisition increment less than 1, configured value is " + bean.getAcquireIncrement());
}
if (xaFactory == null)
{
this.xaFactory = XAFactoryHelper.createXAFactory(bean);
}
else
{
this.xaFactory = xaFactory;
}
init();
if (bean.getIgnoreRecoveryFailures())
{
log.warning("resource '" + bean.getUniqueName() + "' is configured to ignore recovery failures, make sure this setting is not enabled on a production system!");
}
}
/**
* Method init ...
*
* @throws Exception
* when
*/
private void init() throws Exception
{
growUntilMinPoolSize();
if (bean.getMaxIdleTime() > 0 || bean.getMaxLifeTime() > 0)
{
TransactionManagerServices.getTaskScheduler()
.schedulePoolShrinking(this);
}
}
/**
* Method growUntilMinPoolSize ...
*
* @throws Exception
* when
*/
private void growUntilMinPoolSize() throws Exception
{
synchronized (poolGrowthShrinkLock)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("growing " + this + " to minimum pool size " + bean.getMinPoolSize());
}
for (int i = totalPoolSize(); i < bean.getMinPoolSize(); i++)
{
createPooledObject(xaFactory);
}
}
}
/**
* Get the total size of this pool.
*
* @return the total size of this pool
*/
public int totalPoolSize()
{
return poolSize.get();
}
/**
* Method createPooledObject ...
*
* @param xaFactory
* of type Object
*
* @throws Exception
* when
*/
private void createPooledObject(Object xaFactory) throws Exception
{
T xaStatefulHolder = xaResourceProducer.createPooledConnection(xaFactory, bean);
xaStatefulHolder.addStateChangeEventListener(this);
availablePool.addLast(xaStatefulHolder);
poolSize.incrementAndGet();
}
/* -----------------------------------------------------------------------------------
* Pool Transition. stateChanging() and stateChanged() obtain the stateTransitionLock
* write lock to prevent other threads from iterating the pools while a coonection is
* in transition from one pool to another. However, this window of time is extremely
* small, and in general allows high-concurrency.
* ----------------------------------------------------------------------------------*/
/**
* Close down and cleanup this XAPool instance.
*/
public void close()
{
synchronized (poolGrowthShrinkLock)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("closing all connections of " + this);
}
for (T xaStatefulHolder : getXAResourceHolders())
{
try
{
xaStatefulHolder.close();
}
catch (Exception ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "ignoring exception while closing connection " + xaStatefulHolder, ex);
}
}
}
if (TransactionManagerServices.isTaskSchedulerRunning())
{
TransactionManagerServices.getTaskScheduler()
.cancelPoolShrinking(this);
}
stateTransitionLock.writeLock()
.lock();
try
{
availablePool.clear();
accessiblePool.clear();
inaccessiblePool.clear();
failed.set(false);
}
finally
{
stateTransitionLock.writeLock()
.unlock();
}
}
}
/**
* Get a connection handle from this pool.
*
* @param recycle
* true if we should try to get a connection in the NON_ACCESSIBLE pool in the same transaction
*
* @return a connection handle
*
* @throws Exception
* throw in the pool is unrecoverable or a timeout occurs getting a connection
*/
public Object getConnectionHandle(boolean recycle) throws Exception
{
synchronized (poolGrowthShrinkLock)
{
if (isFailed())
{
reinitializePool();
}
}
long remainingTimeMs = TimeUnit.SECONDS.toMillis(bean.getAcquisitionTimeout());
while (true)
{
long before = MonotonicClock.currentTimeMillis();
T xaStatefulHolder = null;
if (recycle)
{
if (bean.getShareTransactionConnections())
{
xaStatefulHolder = getSharedXAStatefulHolder();
}
else
{
xaStatefulHolder = getNotAccessible();
}
}
if (xaStatefulHolder == null)
{
xaStatefulHolder = getInPool(remainingTimeMs);
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("found " + xaStatefulHolder.getState() + " connection " + xaStatefulHolder + " from " + this);
}
try
{
// getConnectionHandle() here could throw an exception, if it doesn't the connection is
// still alive and we can share it (if sharing is enabled)
Object connectionHandle = xaStatefulHolder.getConnectionHandle();
if (bean.getShareTransactionConnections())
{
putSharedXAStatefulHolder(xaStatefulHolder);
}
return connectionHandle;
}
catch (Exception ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "connection is invalid, trying to close it", ex);
}
try
{
xaStatefulHolder.close();
}
catch (Exception ex2)
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "exception while trying to close invalid connection, ignoring it", ex2);
}
}
finally
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("removed invalid connection " + xaStatefulHolder + " from " + this);
}
if (xaStatefulHolder.getState() != State.CLOSED)
{
stateChanged(xaStatefulHolder, xaStatefulHolder.getState(), State.CLOSED);
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("waiting " + bean.getAcquisitionInterval() + "s before trying to acquire a connection again from " + this);
}
long waitTime = TimeUnit.SECONDS.toMillis(bean.getAcquisitionInterval());
if (waitTime > 0)
{
waitFor(waitTime);
}
}
// check for timeout
long now = MonotonicClock.currentTimeMillis();
remainingTimeMs -= (now - before);
if (remainingTimeMs <= 0)
{
throw new BitronixRuntimeException("cannot get valid connection from " + this + " after trying for " + bean.getAcquisitionTimeout() + "s", ex);
}
}
} // while true
}
private synchronized void waitFor(long waitTime)
{
try
{
wait(waitTime);
}
catch (InterruptedException ex2)
{
log.log(Level.FINEST, "Waiting exception", ex2);
}
}
/* ------------------------------------------------------------------------
* Methods to obtain a connection from one of the internal pools.
* ------------------------------------------------------------------------*/
/**
* Get a connection handle from this pool.
*
* @return a connection handle
*
* @throws Exception
* throw in the pool is unrecoverable or a timeout occurs getting a connection
*/
public Object getConnectionHandle() throws Exception
{
return getConnectionHandle(true);
}
/**
* Method stateChanged ...
*
* @param source
* of type T
* @param oldState
* of type State
* @param newState
* of type State
*/
@Override
public void stateChanged(T source, State oldState, State newState)
{
stateTransitionLock.writeLock()
.lock();
try
{
switch (newState)
{
case IN_POOL:
if (LogDebugCheck.isDebugEnabled())
{
log.finer("added " + source + " to the available pool");
}
availablePool.addFirst(source);
break;
case ACCESSIBLE:
if (LogDebugCheck.isDebugEnabled())
{
log.finer("added " + source + " to the accessible pool");
}
accessiblePool.add(source);
break;
case NOT_ACCESSIBLE:
if (LogDebugCheck.isDebugEnabled())
{
log.finer("added " + source + " to the inaccessible pool");
}
inaccessiblePool.add(source);
break;
case CLOSED:
source.removeStateChangeEventListener(this);
poolSize.decrementAndGet();
break;
}
}
finally
{
stateTransitionLock.writeLock()
.unlock();
}
}
/**
* Method stateChanging ...
*
* @param source
* of type T
* @param currentState
* of type State
* @param futureState
* of type State
*/
@Override
public void stateChanging(T source, State currentState, State futureState)
{
stateTransitionLock.writeLock()
.lock();
try
{
switch (currentState)
{
case IN_POOL:
// no-op. calling availablePool.remove(source) here is reduncant because it was
// already removed when availablePool.poll() was called.
break;
case ACCESSIBLE:
if (LogDebugCheck.isDebugEnabled())
{
log.finer("removed " + source + " from the accessible pool");
}
accessiblePool.remove(source);
break;
case NOT_ACCESSIBLE:
if (LogDebugCheck.isDebugEnabled())
{
log.finer("removed " + source + " from the inaccessible pool");
}
inaccessiblePool.remove(source);
break;
case CLOSED:
break;
}
}
finally
{
stateTransitionLock.writeLock()
.unlock();
}
}
/**
* Get an IN_POOL connection. This method blocks for up to remainingTimeMs milliseconds
* for someone to return or create a connection in the available pool. If remainingTimeMs
* expires, an exception is thrown. It does not use stateTransitionLock.readLock() because
* the availablePool [a LinkedBlockingQueue] is already thread safe.
*
* @param remainingTimeMs
* the maximum time to wait for a connection
*
* @return a connection from the available (IN_POOL) pool
*
* @throws Exception
* thrown in no connection is available before the remainingTimeMs time expires
*/
private T getInPool(long remainingTimeMs) throws Exception
{
if (inPoolSize() == 0)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("no more free connections in " + this + ", trying to grow it");
}
grow();
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("getting IN_POOL connection from " + this + ", waiting if necessary");
}
try
{
T xaStatefulHolder = availablePool.pollFirst(remainingTimeMs, TimeUnit.MILLISECONDS);
if (xaStatefulHolder == null)
{
if (TransactionManagerServices.isTransactionManagerRunning())
{
TransactionManagerServices.getTransactionManager()
.dumpTransactionContexts();
}
throw new BitronixRuntimeException("XA pool of resource " + bean.getUniqueName() + " still empty after " + bean.getAcquisitionTimeout() + "s wait time");
}
if (expireStatefulHolder(xaStatefulHolder, false))
{
return getInPool(remainingTimeMs);
}
return xaStatefulHolder;
}
catch (InterruptedException e)
{
throw new BitronixRuntimeException("Interrupted while waiting for IN_POOL connection.", e);
}
}
/* ------------------------------------------------------------------------
* Pool growth and pooled object creation
* ------------------------------------------------------------------------*/
/**
* Get a XAStatefulHolder (connection) from the NOT_ACCESSIBLE pool. This method obtains
* the stateTransitionLock.readLock() which prevents any modification during iteration, but
* allows multiple threads to iterate simultaneously.
*
* @return a connection, or null if there are no connections in the inaccessible pool for the current transaction
*/
private T getNotAccessible()
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("trying to recycle a NOT_ACCESSIBLE connection of " + this);
}
BitronixTransaction transaction = TransactionContextHelper.currentTransaction();
if (transaction == null)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("no current transaction, no connection can be in state NOT_ACCESSIBLE when there is no global transaction context");
}
return null;
}
Uid currentTxGtrid = transaction.getResourceManager()
.getGtrid();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("current transaction GTRID is [" + currentTxGtrid + "]");
}
stateTransitionLock.readLock()
.lock();
try
{
for (T xaStatefulHolder : inaccessiblePool)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("found a connection in NOT_ACCESSIBLE state: " + xaStatefulHolder);
}
if (containsXAResourceHolderMatchingGtrid(xaStatefulHolder, currentTxGtrid))
{
return xaStatefulHolder;
}
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("no NOT_ACCESSIBLE connection enlisted in this transaction");
}
return null;
}
finally
{
stateTransitionLock.readLock()
.unlock();
}
}
/**
* Try to get a shared XAStatefulHolder. This method will either return a shared
* XAStatefulHolder or null
. If there is no current transaction,
* XAStatefulHolder's are not shared. If there is a transaction and there is
* a XAStatefulHolder associated with this thread already, we return that XAStatefulHolder
* (provided it is ACCESSIBLE or NOT_ACCESSIBLE).
*
* @return a shared XAStatefulHolder or null
*/
private T getSharedXAStatefulHolder()
{
BitronixTransaction transaction = TransactionContextHelper.currentTransaction();
if (transaction == null)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("no current transaction, shared connection map will not be used");
}
return null;
}
Uid currentTxGtrid = transaction.getResourceManager()
.getGtrid();
StatefulHolderThreadLocal threadLocal = statefulHolderTransactionMap.get(currentTxGtrid);
if (threadLocal != null)
{
T xaStatefulHolder = threadLocal.get();
// Additional sanity checks...
if (xaStatefulHolder != null &&
xaStatefulHolder.getState() != State.IN_POOL &&
xaStatefulHolder.getState() != State.CLOSED)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("sharing connection " + xaStatefulHolder + " in transaction " + currentTxGtrid);
}
return xaStatefulHolder;
}
}
return null;
}
/**
* Method containsXAResourceHolderMatchingGtrid ...
*
* @param xaStatefulHolder
* of type T
* @param currentTxGtrid
* of type Uid
*
* @return boolean
*/
private boolean containsXAResourceHolderMatchingGtrid(T xaStatefulHolder, Uid currentTxGtrid)
{
List extends XAResourceHolder extends XAResourceHolder>> xaResourceHolders = xaStatefulHolder.getXAResourceHolders();
if (LogDebugCheck.isDebugEnabled())
{
log.finer(xaResourceHolders.size() + " xa resource(s) created by connection in NOT_ACCESSIBLE state: " + xaStatefulHolder);
}
class LocalVisitor
implements XAResourceHolderStateVisitor
{
private boolean found;
/**
* Called when visiting all {@link bitronix.tm.internal.XAResourceHolderState}s.
* @param xaResourceHolderState the currently visited {@link bitronix.tm.internal.XAResourceHolderState}
* @return return true
to continue visitation, false
to stop visitation
*/
@Override
public boolean visit(XAResourceHolderState xaResourceHolderState)
{
// compare GTRIDs
BitronixXid bitronixXid = xaResourceHolderState.getXid();
Uid resourceGtrid = bitronixXid.getGlobalTransactionIdUid();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("NOT_ACCESSIBLE xa resource GTRID: " + resourceGtrid);
}
if (currentTxGtrid.equals(resourceGtrid))
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("NOT_ACCESSIBLE xa resource's GTRID matched this transaction's GTRID, recycling it");
}
found = true;
}
return !found; // continue visitation if not found, stop visitation if found
}
}
for (XAResourceHolder extends XAResourceHolder> xaResourceHolder : xaResourceHolders)
{
LocalVisitor xaResourceHolderStateVisitor = new LocalVisitor();
xaResourceHolder.acceptVisitorForXAResourceHolderStates(currentTxGtrid, xaResourceHolderStateVisitor);
if (xaResourceHolderStateVisitor.found)
{
return true;
}
}
return false;
}
/* ------------------------------------------------------------------------
* Pool shrinking and pooled object expiration.
* ------------------------------------------------------------------------*/
/**
* Grow the pool by "acquire increment" amount up to the max pool size.
*
* @throws Exception
* thrown if creating a pooled objects fails
*/
private void grow() throws Exception
{
synchronized (poolGrowthShrinkLock)
{
long totalPoolSize = totalPoolSize();
if (totalPoolSize < bean.getMaxPoolSize())
{
long increment = bean.getAcquireIncrement();
if (totalPoolSize + increment > bean.getMaxPoolSize())
{
increment = bean.getMaxPoolSize() - totalPoolSize;
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("incrementing " + bean.getUniqueName() + " pool size by " + increment + " unit(s) to reach " + (totalPoolSize() + increment) + " connection(s)");
}
for (int i = 0; i < increment; i++)
{
createPooledObject(xaFactory);
}
}
else
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("pool " + bean.getUniqueName() + " already at max size of " + totalPoolSize() + " connection(s), not growing it");
}
}
if (totalPoolSize() < bean.getMinPoolSize())
{
growUntilMinPoolSize();
}
}
}
/**
* Method shrink ...
*
* @throws Exception
* when
*/
public void shrink() throws Exception
{
synchronized (poolGrowthShrinkLock)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("shrinking " + this);
}
expireOrCloseStatefulHolders(false);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("shrunk " + this);
}
}
}
/**
* Method expireOrCloseStatefulHolders ...
*
* @param forceClose
* of type boolean
*
* @throws Exception
* when
*/
private void expireOrCloseStatefulHolders(boolean forceClose) throws Exception
{
int closed = 0;
int availableSize = availablePool.size();
for (int i = 0; i < availableSize; i++)
{
T xaStatefulHolder = availablePool.pollFirst();
if (xaStatefulHolder == null)
{
break;
}
if (expireStatefulHolder(xaStatefulHolder, forceClose))
{
closed++;
}
else
{
availablePool.addLast(xaStatefulHolder);
}
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("closed " + closed + (forceClose ? " " : " idle ") + "connection(s)");
}
growUntilMinPoolSize();
}
/**
* Method expireStatefulHolder ...
*
* @param xaStatefulHolder
* of type T
* @param forceClose
* of type boolean
*
* @return boolean
*/
private boolean expireStatefulHolder(T xaStatefulHolder, boolean forceClose)
{
long expirationTime = Long.MAX_VALUE;
if (bean.getMaxIdleTime() > 0)
{
expirationTime = xaStatefulHolder.getLastReleaseDate()
.getTime() + TimeUnit.SECONDS.toMillis(bean.getMaxIdleTime());
}
if (bean.getMaxLifeTime() > 0)
{
long endOfLife = xaStatefulHolder.getCreationDate()
.getTime() + TimeUnit.SECONDS.toMillis(bean.getMaxLifeTime());
expirationTime = Math.min(expirationTime, endOfLife);
}
long now = MonotonicClock.currentTimeMillis();
if (!forceClose && LogDebugCheck.isDebugEnabled())
{
log.finer("checking if connection can be closed: " + xaStatefulHolder + " - closing time: " + expirationTime + ", now time: " + now);
}
if (expirationTime <= now || forceClose)
{
try
{
xaStatefulHolder.close();
}
catch (Exception ex)
{
log.log(Level.WARNING, "error closing " + xaStatefulHolder, ex);
}
return true;
}
return false;
}
/**
* Method getNextShrinkDate returns the nextShrinkDate of this XAPool object.
*
* @return the nextShrinkDate (type Date) of this XAPool object.
*/
public Date getNextShrinkDate()
{
return new Date(MonotonicClock.currentTimeMillis() + TimeUnit.SECONDS.toMillis(bean.getMaxIdleTime()));
}
/**
* Method reset ...
*
* @throws Exception
* when
*/
public void reset() throws Exception
{
synchronized (poolGrowthShrinkLock)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("resetting " + this);
}
expireOrCloseStatefulHolders(true);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("reset " + this);
}
}
}
/* ------------------------------------------------------------------------
* Miscellaneous public methods
* ------------------------------------------------------------------------*/
/**
* Method reinitializePool ...
*/
private void reinitializePool()
{
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("resource '" + bean.getUniqueName() + "' is marked as failed, resetting and recovering it before trying connection acquisition");
}
close();
init();
IncrementalRecoverer.recover(xaResourceProducer);
}
catch (RecoveryException ex)
{
throw new BitronixRuntimeException("incremental recovery failed when trying to acquire a connection from failed resource '" + bean.getUniqueName() + "'", ex);
}
catch (Exception ex)
{
throw new BitronixRuntimeException("pool reset failed when trying to acquire a connection from failed resource '" + bean.getUniqueName() + "'", ex);
}
}
/**
* Get the XAFactory (XADataSource) that produces objects for this pool.
*
* @return the factory (XADataSource) object
*/
public Object getXAFactory()
{
return xaFactory;
}
/**
* Method getXAResourceHolders returns the XAResourceHolders of this XAPool object.
*
* @return the XAResourceHolders (type List T ) of this XAPool object.
*/
public List getXAResourceHolders()
{
stateTransitionLock.readLock()
.lock();
try
{
List holders = new ArrayList<>();
holders.addAll(availablePool);
holders.addAll(accessiblePool);
holders.addAll(inaccessiblePool);
return holders;
}
finally
{
stateTransitionLock.readLock()
.unlock();
}
}
/**
* Method toString ...
*
* @return String
*/
@Override
public String toString()
{
return "an XAPool of resource " + bean.getUniqueName() + " with " + totalPoolSize() + " connection(s) (" + inPoolSize() + " still available)" +
(isFailed() ? " -failed-" : "");
}
/**
* Get the number of objects in the available pool.
*
* @return the number of available objects
*/
public int inPoolSize()
{
return availablePool.size();
}
/**
* Is the XAPool in a failed state?
*
* @return true if this XAPool has failed, false otherwise
*/
public boolean isFailed()
{
return failed.get();
}
/**
* Sets this XAPool as failed or unfailed, requiring recovery.
*
* @param failed
* true if this XAPool has failed and requires recovery, false if it is ok
*/
public void setFailed(boolean failed)
{
this.failed.set(failed);
}
/* ------------------------------------------------------------------------
* Shared Connection Handling
* ------------------------------------------------------------------------*/
/**
* Try to share a XAStatefulHolder with other callers on this thread. If
* there is no current transaction, the XAStatefulHolder is not put into the
* ThreadLocal. If there is a transaction, and it is the first time we're
* attempting to share a XAStatefulHolder on this thread, then we register
* a Synchronization so we can pull the ThreadLocals out of the shared map
* when the transaction completes (either commit() or rollback()). Without
* the Synchronization we would "leak".
*
* @param xaStatefulHolder
* a XAStatefulHolder to share with other callers
* on this thread.
*/
private void putSharedXAStatefulHolder(T xaStatefulHolder)
{
BitronixTransaction transaction = TransactionContextHelper.currentTransaction();
if (transaction == null)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("no current transaction, not adding " + xaStatefulHolder + " to shared connection map");
}
return;
}
Uid currentTxGtrid = transaction.getResourceManager()
.getGtrid();
StatefulHolderThreadLocal threadLocal = statefulHolderTransactionMap.get(currentTxGtrid);
if (threadLocal == null)
{
// This is the first time this TxGtrid/ThreadLocal is going into the map,
// register interest in synchronization so we can remove it at commit/rollback
try
{
transaction.registerSynchronization(new SharedStatefulHolderCleanupSynchronization(currentTxGtrid));
}
catch (Exception e)
{
log.log(Level.FINEST, "Ignoreing Exception", e);
// OK, forget it. The transaction is either rollback only or already finished.
return;
}
threadLocal = new StatefulHolderThreadLocal<>();
statefulHolderTransactionMap.put(currentTxGtrid, threadLocal);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("added shared connection mapping for " + currentTxGtrid + " holder " + xaStatefulHolder);
}
}
// Set the XAStatefulHolder on the ThreadLocal. Even if we've already set it before,
// it's safe -- checking would be more expensive than just setting it again.
threadLocal.set(xaStatefulHolder);
}
private static final class StatefulHolderThreadLocal
extends ThreadLocal
{
//Nothing required
}
private final class SharedStatefulHolderCleanupSynchronization
implements Synchronization
{
private final Uid gtrid;
/**
* Constructor SharedStatefulHolderCleanupSynchronization creates a new SharedStatefulHolderCleanupSynchronization instance.
*
* @param gtrid
* of type Uid
*/
private SharedStatefulHolderCleanupSynchronization(Uid gtrid)
{
this.gtrid = gtrid;
}
/**
* Method beforeCompletion ...
*/
@Override
public void beforeCompletion()
{
//Nothing required
}
/**
* Method afterCompletion ...
*
* @param status
* of type int
*/
@Override
public void afterCompletion(int status)
{
statefulHolderTransactionMap.remove(gtrid);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("deleted shared connection mappings for " + gtrid);
}
}
/**
* Method toString ...
*
* @return String
*/
@Override
public String toString()
{
return "a SharedStatefulHolderCleanupSynchronization with GTRID [" + gtrid + "]";
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy