bitronix.tm.resource.jdbc.PoolingDataSource 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.jdbc;
import bitronix.tm.internal.LogDebugCheck;
import bitronix.tm.internal.XAResourceHolderState;
import bitronix.tm.recovery.RecoveryException;
import bitronix.tm.resource.ResourceConfigurationException;
import bitronix.tm.resource.ResourceObjectFactory;
import bitronix.tm.resource.ResourceRegistrar;
import bitronix.tm.resource.common.RecoveryXAResourceHolder;
import bitronix.tm.resource.common.ResourceBean;
import bitronix.tm.resource.common.XAPool;
import bitronix.tm.resource.common.XAResourceProducer;
import bitronix.tm.utils.ManagementRegistrar;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.StringRefAddr;
import javax.sql.DataSource;
import javax.sql.XADataSource;
import javax.transaction.xa.XAResource;
import java.io.PrintWriter;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.logging.Level;
/**
* Implementation of a JDBC {@link DataSource} wrapping vendor's {@link XADataSource} implementation.
*
* @author Ludovic Orban
* @author Brett Wooldridge
*/
@SuppressWarnings("serial")
public class PoolingDataSource
extends ResourceBean
implements DataSource, XAResourceProducer, PoolingDataSourceMBean
{
private static final java.util.logging.Logger log = java.util.logging.Logger.getLogger(PoolingDataSource.class.toString());
private final transient List connectionCustomizers = new CopyOnWriteArrayList<>();
private transient volatile XAPool pool;
private transient volatile XADataSource xaDataSource;
private transient volatile RecoveryXAResourceHolder recoveryXAResourceHolder;
private transient volatile Connection recoveryConnectionHandle;
private transient volatile Map xaResourceHolderMap;
private volatile String testQuery;
private volatile boolean enableJdbc4ConnectionTest;
private volatile int connectionTestTimeout;
private volatile int preparedStatementCacheSize = 0;
private volatile String isolationLevel;
private volatile String cursorHoldability;
private volatile String localAutoCommit;
private volatile String jmxName;
/**
* Initialize all properties with their default values.
*/
public PoolingDataSource()
{
xaResourceHolderMap = new ConcurrentHashMap<>();
}
/**
* @return the wrapped XADataSource.
*/
public XADataSource getXaDataSource()
{
return xaDataSource;
}
/**
* Inject a pre-configured XADataSource instead of relying on className and driverProperties
* to build one. Upon deserialization the xaDataSource will be null and will need to be
* manually re-injected.
*
* @param xaDataSource
* the pre-configured XADataSource.
*/
public void setXaDataSource(XADataSource xaDataSource)
{
this.xaDataSource = xaDataSource;
}
/**
* @return the query that will be used to test connections.
*/
public String getTestQuery()
{
return testQuery;
}
/**
* When set, the specified query will be executed on the connection acquired from the pool before being handed to
* the caller. The connections won't be tested when not set. Default value is null.
*
* @param testQuery
* the query that will be used to test connections.
*/
public void setTestQuery(String testQuery)
{
this.testQuery = testQuery;
}
/**
* @return true if JDBC 4 isValid() testing should be performed, false otherwise.
*/
public boolean isEnableJdbc4ConnectionTest()
{
return enableJdbc4ConnectionTest;
}
/**
* When set and the underlying JDBC driver supports JDBC 4 isValid(), a Connection.isValid() call
* is performed to test the connection before handing it to the caller.
* If both testQuery and enableJdbc4ConnectionTest are set, enableJdbc4ConnectionTest takes precedence.
*
* @param enableJdbc4ConnectionTest
* true if JDBC 4 isValid() testing should be performed, false otherwise.
*/
public void setEnableJdbc4ConnectionTest(boolean enableJdbc4ConnectionTest)
{
this.enableJdbc4ConnectionTest = enableJdbc4ConnectionTest;
}
/**
* @return how many seconds each connection test will wait for a response,
* bounded above by the acquisition timeout.
*/
public int getEffectiveConnectionTestTimeout()
{
int t1 = getConnectionTestTimeout();
int t2 = getAcquisitionTimeout();
if ((t1 > 0) && (t2 > 0))
{
return Math.min(t1, t2);
}
else
{
return Math.max(t1, t2);
}
}
/**
* @return how many seconds each connection test will wait for a response.
*/
public int getConnectionTestTimeout()
{
return connectionTestTimeout;
}
/**
* Determines how many seconds the connection test logic
* will wait for a response from the database.
*
* @param connectionTestTimeout
* connection timeout
*/
public void setConnectionTestTimeout(int connectionTestTimeout)
{
this.connectionTestTimeout = connectionTestTimeout;
}
/**
* @return the target maximum prepared statement cache size.
*/
public int getPreparedStatementCacheSize()
{
return preparedStatementCacheSize;
}
/**
* Set the target maximum size of the prepared statement cache. In
* reality under certain unusual conditions the cache may temporarily
* drift higher in size.
*
* @param preparedStatementCacheSize
* the target maximum prepared statement cache size.
*/
public void setPreparedStatementCacheSize(int preparedStatementCacheSize)
{
this.preparedStatementCacheSize = preparedStatementCacheSize;
}
/**
* @return the default isolation level.
*/
public String getIsolationLevel()
{
return isolationLevel;
}
/**
* Set the default isolation level for connections.
*
* @param isolationLevel
* the default isolation level.
*/
public void setIsolationLevel(String isolationLevel)
{
this.isolationLevel = isolationLevel;
}
/**
* @return cursorHoldability the default cursor holdability.
*/
public String getCursorHoldability()
{
return cursorHoldability;
}
/**
* Set the default cursor holdability for connections.
*
* @param cursorHoldability
* the default cursor holdability.
*/
public void setCursorHoldability(String cursorHoldability)
{
this.cursorHoldability = cursorHoldability;
}
/**
* @return localAutoCommit the default local transactions autocommit mode.
*/
public String getLocalAutoCommit()
{
return localAutoCommit;
}
/**
* Set the default local transactions autocommit mode.
*
* @param localAutoCommit
* the default local transactions autocommit mode.
*/
public void setLocalAutoCommit(String localAutoCommit)
{
this.localAutoCommit = localAutoCommit;
}
/**
* Method addConnectionCustomizer ...
*
* @param connectionCustomizer
* of type ConnectionCustomizer
*/
public void addConnectionCustomizer(ConnectionCustomizer connectionCustomizer)
{
connectionCustomizers.add(connectionCustomizer);
}
/**
* Method removeConnectionCustomizer ...
*
* @param connectionCustomizer
* of type ConnectionCustomizer
*/
public void removeConnectionCustomizer(ConnectionCustomizer connectionCustomizer)
{
Iterator it = connectionCustomizers.iterator();
while (it.hasNext())
{
ConnectionCustomizer customizer = it.next();
if (customizer == connectionCustomizer)
{
it.remove();
return;
}
}
}
/**
* Method fireOnAcquire ...
*
* @param connection
* of type Connection
*/
void fireOnAcquire(Connection connection)
{
for (ConnectionCustomizer connectionCustomizer : connectionCustomizers)
{
try
{
connectionCustomizer.onAcquire(connection, getUniqueName());
}
catch (Exception ex)
{
log.log(Level.WARNING, "ConnectionCustomizer.onAcquire() failed for " + connectionCustomizer, ex);
}
}
}
/**
* Method fireOnLease ...
*
* @param connection
* of type Connection
*/
void fireOnLease(Connection connection)
{
for (ConnectionCustomizer connectionCustomizer : connectionCustomizers)
{
try
{
connectionCustomizer.onLease(connection, getUniqueName());
}
catch (Exception ex)
{
log.log(Level.WARNING, "ConnectionCustomizer.onLease() failed for " + connectionCustomizer, ex);
}
}
}
/**
* Method fireOnRelease ...
*
* @param connection
* of type Connection
*/
void fireOnRelease(Connection connection)
{
for (ConnectionCustomizer connectionCustomizer : connectionCustomizers)
{
try
{
connectionCustomizer.onRelease(connection, getUniqueName());
}
catch (Exception ex)
{
log.log(Level.WARNING, "ConnectionCustomizer.onRelease() failed for " + connectionCustomizer, ex);
}
}
}
/**
* Method fireOnDestroy ...
*
* @param connection
* of type Connection
*/
void fireOnDestroy(Connection connection)
{
for (ConnectionCustomizer connectionCustomizer : connectionCustomizers)
{
try
{
connectionCustomizer.onDestroy(connection, getUniqueName());
}
catch (Exception ex)
{
log.log(Level.WARNING, "ConnectionCustomizer.onDestroy() failed for " + connectionCustomizer, ex);
}
}
}
/**
* Method toString ...
*
* @return String
*/
@Override
public String toString()
{
return "a PoolingDataSource containing " + pool;
}
/**
* Prepare the recoverable {@link javax.transaction.xa.XAResource} producer for recovery.
*
* @return a {@link bitronix.tm.internal.XAResourceHolderState} object that can be used to call recover()
.
*
* @throws bitronix.tm.recovery.RecoveryException
* thrown when a {@link bitronix.tm.internal.XAResourceHolderState} cannot be acquired.
*/
/* XAResourceProducer implementation */
@Override
public XAResourceHolderState startRecovery() throws RecoveryException
{
init();
if (recoveryConnectionHandle != null)
{
throw new RecoveryException("recovery already in progress on " + this);
}
try
{
recoveryConnectionHandle = (Connection) pool.getConnectionHandle(false);
PooledConnectionProxy pooledConnection = (PooledConnectionProxy) recoveryConnectionHandle;
recoveryXAResourceHolder = pooledConnection.getPooledConnection()
.createRecoveryXAResourceHolder();
return new XAResourceHolderState(pooledConnection.getPooledConnection(), this);
}
catch (Exception ex)
{
throw new RecoveryException("cannot start recovery on " + this, ex);
}
}
/**
* Release internal resources held after call to startRecovery()
.
*
* @throws bitronix.tm.recovery.RecoveryException
* thrown when an error occurred while releasing reserved resources.
*/
@Override
public void endRecovery() throws RecoveryException
{
if (recoveryConnectionHandle == null)
{
return;
}
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("recovery xa resource is being closed: " + recoveryXAResourceHolder);
}
recoveryConnectionHandle.close();
}
catch (Exception ex)
{
throw new RecoveryException("error ending recovery on " + this, ex);
}
finally
{
recoveryConnectionHandle = null;
// the recoveryXAResourceHolder actually wraps the recoveryConnectionHandle so closing it
// would close the recoveryConnectionHandle twice which must not happen
recoveryXAResourceHolder = null;
}
} /* Implementation of DataSource interface */
/**
* Method buildXAPool ...
*
* @throws Exception
* when
*/
private void buildXAPool() throws Exception
{
if (pool != null)
{
return;
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("building XA pool for " + getUniqueName() + " with " + getMinPoolSize() + " connection(s)");
}
pool = new XAPool<>(this, this, xaDataSource);
boolean builtXaFactory = false;
if (xaDataSource == null)
{
xaDataSource = (XADataSource) pool.getXAFactory();
builtXaFactory = true;
}
try
{
ResourceRegistrar.register(this);
}
catch (RecoveryException ex)
{
if (builtXaFactory)
{
xaDataSource = null;
}
pool = null;
throw ex;
}
}
/**
* Method unwrap ...
*
* @param iface
* of type Class T
*
* @return T
*
* @throws SQLException
* when
*/
@SuppressWarnings("unchecked")
@Override
public T unwrap(Class iface) throws SQLException
{
if (isWrapperFor(iface))
{
return (T) xaDataSource;
}
throw new SQLException(getClass().getName() + " is not a wrapper for " + iface);
}
/**
* Method isWrapperFor ...
*
* @param iface
* of type Class ?
*
* @return boolean
*
* @throws SQLException
* when
*/
@Override
public boolean isWrapperFor(Class> iface) throws SQLException
{
return iface.isAssignableFrom(xaDataSource.getClass());
}
/**
* Method getInPoolSize returns the inPoolSize of this PoolingDataSource object.
*
* @return the inPoolSize (type int) of this PoolingDataSource object.
*/
@Override
public int getInPoolSize()
{
return pool.inPoolSize();
}
/**
* Method getTotalPoolSize returns the totalPoolSize of this PoolingDataSource object.
*
* @return the totalPoolSize (type int) of this PoolingDataSource object.
*/
@Override
public int getTotalPoolSize()
{
return pool.totalPoolSize();
}
/**
* Method isFailed returns the failed of this PoolingDataSource object.
*
* @return the failed (type boolean) of this PoolingDataSource object.
*/
@Override
public boolean isFailed()
{
return (pool != null ? pool.isFailed() : false);
}
/**
* Mark this resource producer as failed or not. A resource is considered failed if recovery fails to run on it.
*
* @param failed
* true is the resource must be considered failed, false it it must be considered sane.
*/
@Override
public void setFailed(boolean failed)
{
if (pool != null)
{
pool.setFailed(failed);
}
}
/**
* Find in the {@link bitronix.tm.resource.common.XAResourceHolder}s created by this {@link bitronix.tm.resource.common.XAResourceProducer} the one which this
* {@link javax.transaction.xa.XAResource} belongs to.
*
* @param xaResource
* the {@link javax.transaction.xa.XAResource} to look for.
*
* @return the associated {@link bitronix.tm.resource.common.XAResourceHolder} or null if the {@link javax.transaction.xa.XAResource} does not belong to this
* {@link bitronix.tm.resource.common.XAResourceProducer}.
*/
@Override
public JdbcPooledConnection findXAResourceHolder(XAResource xaResource)
{
return xaResourceHolderMap.get(xaResource);
}
/**
* Initializes the pool by creating the initial amount of connections.
*/
@Override
public synchronized void init()
{
if (pool != null)
{
return;
}
try
{
buildXAPool();
jmxName = "bitronix.tm:type=JDBC,UniqueName=" + ManagementRegistrar.makeValidName(getUniqueName());
ManagementRegistrar.register(jmxName, this);
}
catch (Exception ex)
{
throw new ResourceConfigurationException("cannot create JDBC datasource named " + getUniqueName(), ex);
}
}
/**
* Release this {@link bitronix.tm.resource.common.XAResourceProducer}'s internal resources.
*/
@Override
public void close()
{
if (pool == null)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("trying to close already closed PoolingDataSource " + getUniqueName());
}
return;
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("closing " + this);
}
pool.close();
pool = null;
xaResourceHolderMap.clear();
connectionCustomizers.clear();
ManagementRegistrar.unregister(jmxName);
jmxName = null;
ResourceRegistrar.unregister(this);
}
/**
* Create a {@link bitronix.tm.resource.common.XAStatefulHolder} that will be placed in an {@link bitronix.tm.resource.common.XAPool}.
*
* @param xaFactory
* the vendor's resource-specific XA factory.
* @param bean
* the resource-specific bean describing the resource parameters.
*
* @return a {@link bitronix.tm.resource.common.XAStatefulHolder} that will be placed in an {@link bitronix.tm.resource.common.XAPool}.
*
* @throws Exception
* thrown when the {@link bitronix.tm.resource.common.XAStatefulHolder} cannot be created.
*/
@Override
public JdbcPooledConnection createPooledConnection(Object xaFactory, ResourceBean bean) throws Exception
{
if (!(xaFactory instanceof XADataSource))
{
throw new IllegalArgumentException("class '" + xaFactory.getClass()
.getName() + "' does not implement " + XADataSource.class.getName());
}
XADataSource xads = (XADataSource) xaFactory;
JdbcPooledConnection pooledConnection = new JdbcPooledConnection(this, xads.getXAConnection());
xaResourceHolderMap.put(pooledConnection.getXAResource(), pooledConnection);
return pooledConnection;
}
/**
* Method reset ...
*
* @throws Exception
* when
*/
@Override
public void reset() throws Exception
{
pool.reset();
}
/**
* {@link PoolingDataSource} must alway have a unique name so this method builds a reference to this object using
* the unique name as {@link javax.naming.RefAddr}.
*
* @return a reference to this {@link PoolingDataSource}.
*/
@Override
public Reference getReference() throws NamingException
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("creating new JNDI reference of " + this);
}
return new Reference(
PoolingDataSource.class.getName(),
new StringRefAddr("uniqueName", getUniqueName()),
ResourceObjectFactory.class.getName(),
null);
}
/**
* Method unregister ...
*
* @param xaResourceHolder
* of type JdbcPooledConnection
*/
public void unregister(JdbcPooledConnection xaResourceHolder)
{
xaResourceHolderMap.remove(xaResourceHolder.getXAResource());
}
/**
* Method getParentLogger returns the parentLogger of this PoolingDataSource object.
*
* @return the parentLogger (type Logger) of this PoolingDataSource object.
*
* @throws SQLFeatureNotSupportedException
* when
*/
@Override
public java.util.logging.Logger getParentLogger() throws SQLFeatureNotSupportedException
{
throw new SQLFeatureNotSupportedException();
}
/**
* Method getConnection returns the connection of this PoolingDataSource object.
*
* @return the connection (type Connection) of this PoolingDataSource object.
*
* @throws SQLException
* when
*/
@Override
public Connection getConnection() throws SQLException
{
if (isDisabled())
{
throw new SQLException("JDBC connection pool '" + getUniqueName() + "' is disabled, cannot get a connection from it");
}
init();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("acquiring connection from " + this);
}
if (pool == null)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("pool is closed, returning null connection");
}
return null;
}
try
{
Connection conn = (Connection) pool.getConnectionHandle();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("acquired connection from " + this);
}
return conn;
}
catch (Exception ex)
{
throw new SQLException("unable to get a connection from pool of " + this, ex);
}
}
/**
* Method getConnection ...
*
* @param username
* of type String
* @param password
* of type String
*
* @return Connection
*
* @throws SQLException
* when
*/
@Override
public Connection getConnection(String username, String password) throws SQLException
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("JDBC connections are pooled, username and password ignored");
}
return getConnection();
}
/* DataSource implementation */
/**
* Method getLoginTimeout returns the loginTimeout of this PoolingDataSource object.
*
* @return the loginTimeout (type int) of this PoolingDataSource object.
*
* @throws SQLException
* when
*/
@Override
public int getLoginTimeout() throws SQLException
{
return xaDataSource.getLoginTimeout();
}
/**
* Method setLoginTimeout sets the loginTimeout of this PoolingDataSource object.
*
* @param seconds
* the loginTimeout of this PoolingDataSource object.
*
* @throws SQLException
* when
*/
@Override
public void setLoginTimeout(int seconds) throws SQLException
{
xaDataSource.setLoginTimeout(seconds);
}
/**
* Method getLogWriter returns the logWriter of this PoolingDataSource object.
*
* @return the logWriter (type PrintWriter) of this PoolingDataSource object.
*
* @throws SQLException
* when
*/
@Override
public PrintWriter getLogWriter() throws SQLException
{
return xaDataSource.getLogWriter();
}
/**
* Method setLogWriter sets the logWriter of this PoolingDataSource object.
*
* @param out
* the logWriter of this PoolingDataSource object.
*
* @throws SQLException
* when
*/
@Override
public void setLogWriter(PrintWriter out) throws SQLException
{
xaDataSource.setLogWriter(out);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy