com.atomikos.datasource.pool.ConnectionPool Maven / Gradle / Ivy
/**
* Copyright (C) 2000-2013 Atomikos
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* 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.
*/
package com.atomikos.datasource.pool;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import com.atomikos.icatch.HeuristicMessage;
import com.atomikos.icatch.imp.thread.InterruptedExceptionHelper;
import com.atomikos.icatch.imp.thread.TaskManager;
import com.atomikos.logging.Logger;
import com.atomikos.logging.LoggerFactory;
import com.atomikos.timing.AlarmTimer;
import com.atomikos.timing.AlarmTimerListener;
import com.atomikos.timing.PooledAlarmTimer;
public class ConnectionPool implements XPooledConnectionEventListener
{
private static final Logger LOGGER = LoggerFactory.createLogger(ConnectionPool.class);
public final static int DEFAULT_MAINTENANCE_INTERVAL = 60;
private List connections = new ArrayList();
private ConnectionFactory connectionFactory;
private ConnectionPoolProperties properties;
private boolean destroyed;
private PooledAlarmTimer maintenanceTimer;
private String name;
public ConnectionPool ( ConnectionFactory connectionFactory , ConnectionPoolProperties properties ) throws ConnectionPoolException
{
this.connectionFactory = connectionFactory;
this.properties = properties;
this.destroyed = false;
this.name = properties.getUniqueResourceName();
init();
}
private void assertNotDestroyed() throws ConnectionPoolException
{
if (destroyed) throw new ConnectionPoolException ( "Pool was already destroyed - you can no longer use it" );
}
private void init() throws ConnectionPoolException
{
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": initializing..." );
addConnectionsIfMinPoolSizeNotReached();
launchMaintenanceTimer();
}
private void launchMaintenanceTimer() {
int maintenanceInterval = properties.getMaintenanceInterval();
if ( maintenanceInterval <= 0 ) {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": using default maintenance interval..." );
maintenanceInterval = DEFAULT_MAINTENANCE_INTERVAL;
}
maintenanceTimer = new PooledAlarmTimer ( maintenanceInterval * 1000 );
maintenanceTimer.addAlarmTimerListener(new AlarmTimerListener() {
public void alarm(AlarmTimer timer) {
reapPool();
removeConnectionsThatExceededMaxLifetime();
addConnectionsIfMinPoolSizeNotReached();
removeIdleConnectionsIfMinPoolSizeExceeded();
}
});
TaskManager.getInstance().executeTask ( maintenanceTimer );
}
private synchronized void addConnectionsIfMinPoolSizeNotReached() {
int connectionsToAdd = properties.getMinPoolSize() - totalSize();
for ( int i = 0 ; i < connectionsToAdd ; i++ ) {
try {
XPooledConnection xpc = connectionFactory.createPooledConnection();
connections.add ( xpc );
xpc.registerXPooledConnectionEventListener ( this );
} catch ( Exception dbDown ) {
//see case 26380
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": could not establish initial connection" , dbDown );
}
}
}
private Reapable recycleConnectionIfPossible ( HeuristicMessage hmsg ) throws Exception
{
Reapable ret = null;
for (int i = 0; i < totalSize(); i++) {
XPooledConnection xpc = (XPooledConnection) connections.get(i);
if (xpc.canBeRecycledForCallingThread()) {
ret = xpc.createConnectionProxy ( hmsg );
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug( this + ": recycling connection from pool..." );
return ret;
}
}
return ret;
}
/**
* Borrows a connection from the pool.
* @param hmsg The heuristic message to get the connection with.
* @return The connection as Reapable.
* @throws CreateConnectionException If the pool attempted to grow but failed.
* @throws PoolExhaustedException If the pool could not grow because it is exhausted.
* @throws ConnectionPoolException Other errors.
*/
public synchronized Reapable borrowConnection ( HeuristicMessage hmsg ) throws CreateConnectionException , PoolExhaustedException, ConnectionPoolException
{
assertNotDestroyed();
Reapable ret = null;
ret = findExistingOpenConnectionForCallingThread(hmsg);
if (ret == null) {
ret = findOrWaitForAnAvailableConnection(hmsg);
}
return ret;
}
private Reapable findOrWaitForAnAvailableConnection(HeuristicMessage hmsg) throws ConnectionPoolException {
Reapable ret = null;
long remainingTime = properties.getBorrowConnectionTimeout() * 1000L;
do {
ret = retrieveFirstAvailableConnectionAndGrowPoolIfNecessary(hmsg);
if ( ret == null ) {
remainingTime = waitForAtLeastOneAvailableConnection(remainingTime);
assertNotDestroyed();
}
} while ( ret == null );
return ret;
}
private Reapable retrieveFirstAvailableConnectionAndGrowPoolIfNecessary(
HeuristicMessage hmsg) throws CreateConnectionException {
Reapable ret = retrieveFirstAvailableConnection(hmsg);
if ( ret == null && canGrow() ) {
growPool();
ret = retrieveFirstAvailableConnection(hmsg);
}
return ret;
}
private Reapable findExistingOpenConnectionForCallingThread(HeuristicMessage hmsg) {
Reapable recycledConnection = null ;
try {
recycledConnection = recycleConnectionIfPossible ( hmsg );
} catch (Exception e) {
//ignore but log
LOGGER.logWarning ( this + ": error while trying to recycle" , e );
}
return recycledConnection;
}
private void logCurrentPoolSize() {
if ( LOGGER.isDebugEnabled() ) {
LOGGER.logDebug( this + ": current size: " + availableSize() + "/" + totalSize());
}
}
private boolean canGrow() {
return totalSize() < properties.getMaxPoolSize();
}
private Reapable retrieveFirstAvailableConnection(HeuristicMessage hmsg) {
Reapable ret = null;
Iterator it = connections.iterator();
while ( it.hasNext() && ret == null ) {
XPooledConnection xpc = it.next();
if (xpc.isAvailable()) {
try {
ret = xpc.createConnectionProxy ( hmsg );
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug( this + ": got connection from pool");
} catch ( CreateConnectionException ex ) {
String msg = this + ": error creating proxy of connection " + xpc;
LOGGER.logWarning( msg , ex);
it.remove();
xpc.destroy();
} finally {
logCurrentPoolSize();
}
}
}
return ret;
}
private synchronized void growPool() throws CreateConnectionException {
XPooledConnection xpc = connectionFactory.createPooledConnection();
connections.add ( xpc );
xpc.registerXPooledConnectionEventListener(this);
logCurrentPoolSize();
}
private synchronized void removeIdleConnectionsIfMinPoolSizeExceeded() {
if (connections == null || properties.getMaxIdleTime() <= 0 )
return;
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": trying to shrink pool" );
List connectionsToRemove = new ArrayList();
int maxConnectionsToRemove = totalSize() - properties.getMinPoolSize();
if ( maxConnectionsToRemove > 0 ) {
for ( int i=0 ; i < connections.size() ; i++ ) {
XPooledConnection xpc = ( XPooledConnection ) connections.get(i);
long lastRelease = xpc.getLastTimeReleased();
long maxIdle = properties.getMaxIdleTime();
long now = System.currentTimeMillis();
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": connection idle for " + (now - lastRelease) + "ms");
if ( xpc.isAvailable() && ( (now - lastRelease) >= (maxIdle * 1000L) ) && ( connectionsToRemove.size() < maxConnectionsToRemove ) ) {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": connection idle for more than " + maxIdle + "s, closing it: " + xpc);
xpc.destroy();
connectionsToRemove.add(xpc);
}
}
}
connections.removeAll(connectionsToRemove);
logCurrentPoolSize();
}
private synchronized void reapPool()
{
long maxInUseTime = properties.getReapTimeout();
if ( connections == null || maxInUseTime <= 0 ) return;
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": reaping old connections" );
Iterator it = connections.iterator();
while ( it.hasNext() ) {
XPooledConnection xpc = it.next();
long lastTimeReleased = xpc.getLastTimeAcquired();
boolean inUse = !xpc.isAvailable();
long now = System.currentTimeMillis();
if ( inUse && ( ( now - maxInUseTime * 1000 ) > lastTimeReleased ) ) {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": connection in use for more than " + maxInUseTime + "s, reaping it: " + xpc );
xpc.reap();
it.remove();
}
}
logCurrentPoolSize();
}
private synchronized void removeConnectionsThatExceededMaxLifetime()
{
long maxLifetime = properties.getMaxLifetime();
if ( connections == null || maxLifetime <= 0 ) return;
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": closing connections that exceeded maxLifetime" );
Iterator it = connections.iterator();
while ( it.hasNext() ) {
XPooledConnection xpc = it.next();
long creationTime = xpc.getCreationTime();
long now = System.currentTimeMillis();
if ( xpc.isAvailable() && ( (now - creationTime) >= (maxLifetime * 1000L) ) ) {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": connection in use for more than " + maxLifetime + "s, destroying it: " + xpc );
xpc.destroy();
it.remove();
}
}
logCurrentPoolSize();
}
public synchronized void destroy()
{
if ( ! destroyed ) {
LOGGER.logInfo ( this + ": destroying pool..." );
for ( int i=0 ; i < connections.size() ; i++ ) {
XPooledConnection xpc = ( XPooledConnection ) connections.get(i);
if ( !xpc.isAvailable() ) {
LOGGER.logWarning ( this + ": connection is still in use on pool destroy: " + xpc +
" - please check your shutdown sequence to avoid heuristic termination " +
"of ongoing transactions!" );
}
xpc.destroy();
}
connections = null;
destroyed = true;
maintenanceTimer.stop();
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": pool destroyed." );
}
}
/**
* Wait until the connection pool contains an available connection or a timeout happens.
* Returns immediately if the pool already contains a connection in state available.
* @throws CreateConnectionException if a timeout happened while waiting for a connection
*/
private synchronized long waitForAtLeastOneAvailableConnection(long waitTime) throws PoolExhaustedException
{
while (availableSize() == 0) {
if ( waitTime <= 0 ) throw new PoolExhaustedException ( "ConnectionPool: pool is empty - increase either maxPoolSize or borrowConnectionTimeout" );
long before = System.currentTimeMillis();
try {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": about to wait for connection during " + waitTime + "ms...");
this.wait (waitTime);
} catch (InterruptedException ex) {
// cf bug 67457
InterruptedExceptionHelper.handleInterruptedException ( ex );
// ignore
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": interrupted during wait" , ex );
}
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug ( this + ": done waiting." );
long now = System.currentTimeMillis();
waitTime -= (now - before);
}
return waitTime;
}
/**
* The amount of pooled connections in state available.
* @return the amount of pooled connections in state available.
*/
public synchronized int availableSize()
{
int ret = 0;
if ( !destroyed ) {
int count = 0;
for ( int i=0 ; i < connections.size() ; i++ ) {
XPooledConnection xpc = (XPooledConnection) connections.get(i);
if (xpc.isAvailable()) count++;
}
ret = count;
}
return ret;
}
/**
* The total amount of pooled connections in any state.
* @return the total amount of pooled connections in any state
*/
public synchronized int totalSize()
{
if ( destroyed ) return 0;
return connections.size();
}
public synchronized void onXPooledConnectionTerminated(XPooledConnection connection) {
if ( LOGGER.isDebugEnabled() ) LOGGER.logDebug( this + ": connection " + connection + " became available, notifying potentially waiting threads");
this.notify();
}
public String toString() {
return "atomikos connection pool '" + name + "'";
}
}