com.atomikos.datasource.pool.AbstractXPooledConnection Maven / Gradle / Ivy
/**
* Copyright (C) 2000-2019 Atomikos
*
* LICENSE CONDITIONS
*
* See http://www.atomikos.com/Main/WhichLicenseApplies for details.
*/
package com.atomikos.datasource.pool;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;
import com.atomikos.logging.Logger;
import com.atomikos.logging.LoggerFactory;
import com.atomikos.logging.StackTrace;
/**
*
*
* Abstract superclass with generic support for XPooledConnection.
*
*
*/
public abstract class AbstractXPooledConnection implements XPooledConnection {
private static final Logger LOGGER = LoggerFactory.createLogger(AbstractXPooledConnection.class);
private static boolean collectStackTraceForNextReap = false;
private long lastTimeAcquired = System.currentTimeMillis();
private long lastTimeReleased = System.currentTimeMillis();
private List> poolEventListeners = new ArrayList>();
private ConnectionType currentProxy = null;
private ConnectionPoolProperties props;
private long creationTime = System.currentTimeMillis();
private final AtomicBoolean isConcurrentlyBeingAcquired = new AtomicBoolean(false);
private final long maxLifetimeInMillis;
private StackTraceElement[] stackTrace;
protected AbstractXPooledConnection ( ConnectionPoolProperties props )
{
this.props = props;
this.maxLifetimeInMillis = props.getMaxLifetime()*1000;
}
protected void processStackTrace() {
if (stackTrace != null) {
LOGGER.logWarning ( this + ": reaping connection - see stacktrace below for how the connection was last acquired (if there is a connection leak then this may help you find it in your application-specific part of this stack trace)" );
LOGGER.logWarning(StackTrace.toString(stackTrace));
} else {
LOGGER.logWarning ( this + ": reaping connection (and starting to collect stack traces for next reap)...");
AbstractXPooledConnection.collectStackTraceForNextReap = true;
}
}
public long getLastTimeAcquired() {
return lastTimeAcquired;
}
public long getLastTimeReleased() {
return lastTimeReleased;
}
public synchronized ConnectionType createConnectionProxy() throws CreateConnectionException
{
if (collectStackTraceForNextReap) {
stackTrace = Thread.currentThread().getStackTrace();
}
updateLastTimeAcquired();
testUnderlyingConnection();
currentProxy = doCreateConnectionProxy();
isConcurrentlyBeingAcquired.set(false);
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": returning proxy " + currentProxy );
return currentProxy;
}
public void registerXPooledConnectionEventListener(XPooledConnectionEventListener listener) {
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": registering listener " + listener );
poolEventListeners.add(listener);
}
public void unregisterXPooledConnectionEventListener(XPooledConnectionEventListener listener) {
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": unregistering listener " + listener );
poolEventListeners.remove(listener);
}
protected void fireOnXPooledConnectionTerminated() {
for (int i=0; i listener = poolEventListeners.get(i);
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": notifying listener: " + listener );
listener.onXPooledConnectionTerminated(this);
}
updateLastTimeReleased();
}
protected String getTestQuery()
{
return props.getTestQuery();
}
protected void updateLastTimeReleased() {
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": updating last time released" );
lastTimeReleased = System.currentTimeMillis();
}
private void updateLastTimeAcquired() {
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": updating last time acquired" );
lastTimeAcquired = System.currentTimeMillis();
}
protected ConnectionType getCurrentConnectionProxy() {
return currentProxy;
}
public boolean canBeRecycledForCallingThread ()
{
//default is false
return false;
}
protected int getDefaultIsolationLevel()
{
return props.getDefaultIsolationLevel();
}
protected int getBorrowConnectionTimeout() {
return props.getBorrowConnectionTimeout();
}
public long getCreationTime() {
return creationTime;
}
public boolean markAsBeingAcquiredIfAvailable() {
synchronized (isConcurrentlyBeingAcquired) {
if (isConcurrentlyBeingAcquired.get()) {
return false;
}
isConcurrentlyBeingAcquired.set(isAvailable());
return isConcurrentlyBeingAcquired.get();
}
}
protected abstract ConnectionType doCreateConnectionProxy() throws CreateConnectionException;
protected abstract void testUnderlyingConnection() throws CreateConnectionException;
protected boolean maxLifetimeExceeded() {
boolean ret = false;
if (maxLifetimeInMillis>0) {
ret = getCreationTime() + maxLifetimeInMillis < System.currentTimeMillis();
}
return ret;
}
public synchronized void destroy(boolean reap) {
if (reap) {
processStackTrace();
} else if (!isAvailable()) {
return; // cf case 172524
}
if ( LOGGER.isTraceEnabled() ) LOGGER.logTrace ( this + ": destroying..." );
doDestroy(reap);
}
protected abstract void doDestroy(boolean reap);
}