- Introduction
- How to use
- Attributes
- Advanced usage
- JDBC interceptors
- Configuring JDBC interceptors
- org.apache.tomcat.jdbc.pool.JdbcInterceptor
- org.apache.tomcat.jdbc.pool.interceptor.ConnectionState
- org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer
- org.apache.tomcat.jdbc.pool.interceptor.StatementCache
- org.apache.tomcat.jdbc.pool.interceptor.StatementDecoratorInterceptor
- org.apache.tomcat.jdbc.pool.interceptor.QueryTimeoutInterceptor
- org.apache.tomcat.jdbc.pool.interceptor.SlowQueryReport
- org.apache.tomcat.jdbc.pool.interceptor.SlowQueryReportJmx
- org.apache.tomcat.jdbc.pool.interceptor.ResetAbandonedTimer
- Code Example
- Building
The Tomcat JDBC Connection Pool
Table of Contents
Introduction
The JDBC Connection Pool org.apache.tomcat.jdbc.pool
is a replacement or an alternative to the Apache
Commons DBCP
connection pool.
So why do we need a new connection pool?
Here are a few of the reasons:
- Commons DBCP 1.x is single threaded. In order to be thread safe Commons locks the entire pool for short periods during both object allocation and object return. Note that this does not apply to Commons DBCP 2.x.
- Commons DBCP 1.x can be slow. As the number of logical CPUs grows and the number of concurrent threads attempting to borrow or return objects increases, the performance suffers. For highly concurrent systems the impact can be significant. Note that this does not apply to Commons DBCP 2.x.
- Commons DBCP is over 60 classes. tomcat-jdbc-pool core is 8 classes, hence modifications for future requirement will require much less changes. This is all you need to run the connection pool itself, the rest is gravy.
- Commons DBCP uses static interfaces. This means you have to use the
right version for a given JRE version or you may see
NoSuchMethodException
exceptions. - It's not worth rewriting over 60 classes, when a connection pool can be accomplished with a much simpler implementation.
- Tomcat jdbc pool implements the ability retrieve a connection asynchronously, without adding additional threads to the library itself.
- Tomcat jdbc pool is a Tomcat module, it depends on Tomcat JULI, a simplified logging framework used in Tomcat.
- Retrieve the underlying connection using the
javax.sql.PooledConnection
interface. - Starvation proof. If a pool is empty, and threads are waiting for a connection, when a connection is returned, the pool will awake the correct thread waiting. Most pools will simply starve.
Features added over other connection pool implementations
- Support for highly concurrent environments and multi core/cpu systems.
- Dynamic implementation of interface, will support
java.sql
andjavax.sql
interfaces for your runtime environment (as long as your JDBC driver does the same), even when compiled with a lower version of the JDK. - Validation intervals - we don't have to validate every single time we use the connection, we can do this when we borrow or return the connection, just not more frequent than an interval we can configure.
- Run-Once query, a configurable query that will be run only once, when the connection to the database is established. Very useful to setup session settings, that you want to exist during the entire time the connection is established.
- Ability to configure custom interceptors.
This allows you to write custom interceptors to enhance the functionality. You can use
interceptors to gather query stats,
cache session states, reconnect the connection upon failures, retry queries, cache query
results, and so on.
Your options are endless and the interceptors are dynamic, not tied to a JDK version of
a
java.sql
/javax.sql
interface. - High performance - we will show some differences in performance later on
- Extremely simple, due to the very simplified implementation, the line count and source file count are very low, compare with c3p0 that has over 200 source files(last time we checked), Tomcat jdbc has a core of 8 files, the connection pool itself is about half that. As bugs may occur, they will be faster to track down, and easier to fix. Complexity reduction has been a focus from inception.
- Asynchronous connection retrieval - you can queue your request for a connection and
receive a
Future<Connection>
back. - Better idle connection handling. Instead of closing connections directly, it can still pool connections and sizes the idle pool with a smarter algorithm.
- You can decide at what moment connections are considered abandoned, is it when the pool is full, or directly at a timeout by specifying a pool usage threshold.
- The abandon connection timer will reset upon a statement/query activity. Allowing a
connections that is in use for a long time to not timeout.
This is achieved using the
ResetAbandonedTimer
- Close connections after they have been connected for a certain time. Age based close upon return to the pool.
- Get JMX notifications and log entries when connections are suspected for being
abandoned. This is similar to
the
removeAbandonedTimeout
but it doesn't take any action, only reports the information. This is achieved using thesuspectTimeout
attribute. - Connections can be retrieved from a
java.sql.Driver
,javax.sql.DataSource
orjavax.sql.XADataSource
This is achieved using thedataSource
anddataSourceJNDI
attributes. - XA connection support
How to use
Usage of the Tomcat connection pool has been made to be as simple as possible, for those of you that are familiar with commons-dbcp, the transition will be very simple. Moving from other connection pools is also fairly straight forward.
Additional features
The Tomcat connection pool offers a few additional features over what most other pools let you do:
initSQL
- the ability to run a SQL statement exactly once, when the connection is createdvalidationInterval
- in addition to running validations on connections, avoid running them too frequently.jdbcInterceptors
- flexible and pluggable interceptors to create any customizations around the pool, the query execution and the result set handling. More on this in the advanced section.fairQueue
- Set the fair flag to true to achieve thread fairness or to use asynchronous connection retrieval
Inside the Apache Tomcat Container
The Tomcat Connection pool is configured as a resource described in The Tomcat JDBC documentation
With the only difference being that you have to specify the factory
attribute and set the value to
org.apache.tomcat.jdbc.pool.DataSourceFactory
Standalone
The connection pool only has another dependency, and that is on tomcat-juli.jar.
To configure the pool in a stand alone project using bean instantiation, the bean to
instantiate is
org.apache.tomcat.jdbc.pool.DataSource
. The same attributes (documented
below) as you use to configure a connection
pool as a JNDI resource, are used to configure a data source as a bean.
JMX
The connection pool object exposes an MBean that can be registered.
In order for the connection pool object to create the MBean, the flag jmxEnabled
has to be set to true.
This doesn't imply that the pool will be registered with an MBean server, merely
that the MBean is created.
In a container like Tomcat, Tomcat itself registers the DataSource with the MBean
server, the
org.apache.tomcat.jdbc.pool.DataSource
object will then register the
actual
connection pool MBean.
If you're running outside of a container, you can register the DataSource yourself
under any object name you specify,
and it propagates the registration to the underlying pool. To do this you would call
mBeanServer.registerMBean(dataSource.getPool().getJmxPool(),objectname)
.
Prior to this call, ensure that the pool has been created by calling dataSource.createPool()
.
Attributes
To provide a very simple switch to and from commons-dbcp and tomcat-jdbc-pool, Most attributes are the same and have the same meaning.
JNDI Factory and Type
Attribute | Description |
---|---|
factory |
factory is required, and the value should be |
type |
Type should always be Depending on the type a
|
System Properties
System properties are JVM wide, affect all pools created in the JVM
Attribute | Description |
---|---|
org.apache.tomcat.jdbc.pool.onlyAttemptCurrentClassLoader
|
(boolean) Controls classloading of dynamic classes, such as
JDBC drivers, interceptors and validators. If set to
|
Common Attributes
These attributes are shared between commons-dbcp and tomcat-jdbc-pool, in some cases default values are different.
Attribute | Description |
---|---|
defaultAutoCommit |
(boolean) The default auto-commit state of connections created by this
pool. If not set, default is JDBC driver default (If not set then the
|
defaultReadOnly |
(boolean) The default read-only state of connections created by this
pool. If not set then the |
defaultTransactionIsolation |
(String) The default TransactionIsolation state of connections created by this pool. One of the following: (see javadoc )
If not set, the method will not be called and it defaults to the JDBC driver. |
defaultCatalog |
(String) The default catalog of connections created by this pool. |
driverClassName |
(String) The fully qualified Java class name of the JDBC driver to be used. The driver has to be accessible from the same classloader as tomcat-jdbc.jar |
username |
(String) The connection username to be passed to our JDBC driver to
establish a connection.
Note that method
|
password |
(String) The connection password to be passed to our JDBC driver to
establish a connection.
Note that method
|
maxActive |
(int) The maximum number of active connections that can be allocated from
this pool at the same time.
The default value is |
maxIdle |
(int) The maximum number of connections that should be kept in the pool
at all times.
Default value is |
minIdle |
(int) The minimum number of established connections that should be kept
in the pool at all times.
The connection pool can shrink below this number if validation queries
fail.
Default value is derived from |
initialSize |
(int)The initial number of connections that are created when the pool is
started.
Default value is |
maxWait |
(int) The maximum number of milliseconds that the pool will wait (when
there are no available connections)
for a connection to be returned before throwing an exception.
Default value is |
testOnBorrow |
(boolean) The indication of whether objects will be validated before
being borrowed from the pool.
If the object fails to validate, it will be dropped from the pool, and
we will attempt to borrow another.
In order to have a more efficient validation, see |
testOnConnect |
(boolean) The indication of whether objects will be validated when a
connection is first created.
If an object fails to validate, it will be throw
|
testOnReturn |
(boolean) The indication of whether objects will be validated before
being returned to the pool.
The default value is |
testWhileIdle |
(boolean) The indication of whether objects will be validated by the idle
object evictor (if any).
If an object fails to validate, it will be dropped from the pool.
The default value is |
validationQuery |
(String) The SQL query that will be used to validate connections from
this pool before returning them to the caller.
If specified, this query does not have to return any data, it just can't
throw a |
validationQueryTimeout |
(int) The timeout in seconds before a connection validation queries fail.
This works by calling
|
validatorClassName |
(String) The name of a class which implements the
|
timeBetweenEvictionRunsMillis |
(int) The number of milliseconds to sleep between runs of the idle
connection validation/cleaner thread.
This value should not be set under 1 second. It dictates how often we
check for idle, abandoned connections, and how often
we validate idle connections.
The default value is |
numTestsPerEvictionRun |
(int) Property not used in tomcat-jdbc-pool. |
minEvictableIdleTimeMillis |
(int) The minimum amount of time an object may sit idle in the pool
before it is eligible for eviction.
The default value is |
accessToUnderlyingConnectionAllowed |
(boolean) Property not used. Access can be achieved by calling |
removeAbandoned |
(boolean) Flag to remove abandoned connections if they exceed the |
removeAbandonedTimeout |
(int) Timeout in seconds before an abandoned(in use) connection can be
removed.
The default value is |
logAbandoned |
(boolean) Flag to log stack traces for application code which abandoned a
Connection.
Logging of abandoned Connections adds overhead for every Connection
borrow because a stack trace has to be generated.
The default value is |
connectionProperties |
(String) The connection properties that will be sent to our JDBC driver
when establishing new connections.
Format of the string must be [propertyName=property;]*
NOTE - The "user" and "password" properties will be passed explicitly,
so they do not need to be included here.
The default value is |
poolPreparedStatements |
(boolean) Property not used. |
maxOpenPreparedStatements |
(int) Property not used. |
Tomcat JDBC Enhanced Attributes
Attribute | Description |
---|---|
initSQL |
(String) A custom query to be run when a connection is first created.
The default value is |
jdbcInterceptors |
(String) A semicolon separated list of classnames extending
These interceptors will be inserted as an interceptor into the chain
of operations on a
Predefined interceptors: More predefined interceptors are described in detail in the JDBC Interceptors section. |
validationInterval |
(long) avoid excess validation, only run validation at most at this
frequency - time in milliseconds.
If a connection is due for validation, but has been validated previously
within this interval, it will not be validated again.
The default value is |
jmxEnabled |
(boolean) Register the pool with JMX or not.
The default value is |
fairQueue |
(boolean) Set to true if you wish that calls to getConnection should be
treated
fairly in a true FIFO fashion. This uses the |
abandonWhenPercentageFull |
(int) Connections that have been abandoned (timed out) wont get closed
and reported up unless
the number of connections in use are above the percentage defined by
|
maxAge |
(long) Time in milliseconds to keep this connection. This attribute
works both when returning connection and when borrowing connection.
When a connection is borrowed from the pool, the pool will check to see
if the |
useEquals |
(boolean) Set to true if you wish the |
suspectTimeout |
(int) Timeout value in seconds. Default value is |
rollbackOnReturn |
(boolean) If |
commitOnReturn |
(boolean) If |
alternateUsernameAllowed |
(boolean) By default, the jdbc-pool will ignore the
The pool can however be configured to allow use of different credentials
each time a connection is requested. To enable the functionality
described in the
|
dataSource |
(javax.sql.DataSource) Inject a data source to the connection pool, and
the pool will use the data source to retrieve connections instead of
establishing them using the |
dataSourceJNDI |
(String) The JNDI name for a data source to be looked up in JNDI and then
used to establish connections to the database. See the
|
useDisposableConnectionFacade |
(boolean) Set this to true if you wish to put a facade on your connection
so that it cannot be reused after it has been closed. This prevents a
thread holding on to a
reference of a connection it has already called closed on, to execute
queries on it. Default value is |
logValidationErrors |
(boolean) Set this to true to log errors during the validation phase to
the log file. If set to true, errors will be logged as SEVERE. Default
value is |
propagateInterruptState |
(boolean) Set this to true to propagate the interrupt state for a thread
that has been interrupted (not clearing the interrupt state). Default
value is |
ignoreExceptionOnPreLoad |
(boolean) Flag whether ignore error of connection creation while
initializing the pool.
Set to true if you want to ignore error of connection creation while
initializing the pool.
Set to false if you want to fail the initialization of the pool by
throwing exception.
The default value is |
useStatementFacade |
(boolean) Set this to true if you wish to wrap statements in order to
enable |
Advanced usage
JDBC interceptors
To see an example of how to use an interceptor, take a look at
org.apache.tomcat.jdbc.pool.interceptor.ConnectionState
.
This simple interceptor is a cache of three attributes, transaction isolation level,
auto commit and read only state,
in order for the system to avoid not needed roundtrips to the database.
Further interceptors will be added to the core of the pool as the need arises. Contributions are always welcome!
Interceptors are of course not limited to just java.sql.Connection
but
can be used to wrap any
of the results from a method invocation as well. You could build query performance
analyzer that provides JMX notifications when a
query is running longer than the expected time.
Configuring JDBC interceptors
Configuring JDBC interceptors is done using the jdbcInterceptors property.
The property contains a list of semicolon separated class names. If the
classname is not fully qualified it will be prefixed with the
org.apache.tomcat.jdbc.pool.interceptor.
prefix.
Example:
jdbcInterceptors="org.apache.tomcat.jdbc.pool.interceptor.ConnectionState;
org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer"
is the same as
jdbcInterceptors="ConnectionState;StatementFinalizer"
Interceptors can have properties as well. Properties for an interceptor are specified within parentheses after the class name. Several properties are separated by commas.
Example:
jdbcInterceptors="ConnectionState;StatementFinalizer(useEquals=true)"
Extra whitespace characters around class names, property names and values are ignored.
org.apache.tomcat.jdbc.pool.JdbcInterceptor
Abstract base class for all interceptors, cannot be instantiated.
Attribute | Description |
---|---|
useEquals |
(boolean) Set to true if you wish the |
org.apache.tomcat.jdbc.pool.interceptor.ConnectionState
Caches the connection for the following attributes autoCommit
, readOnly
,
transactionIsolation
and catalog
.
It is a performance enhancement to avoid roundtrip to the database when getters are
called or setters are called with an already set value.
Attribute | Description |
---|
org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer
Keeps track of all statements created using createStatement
, prepareStatement
or prepareCall
and closes these statements when the connection is returned to the pool.
Attribute | Description |
---|---|
trace |
(boolean as String) Enable tracing of unclosed statements.
When enabled and a connection is closed, and statements are not closed,
the interceptor will log all stack traces.
The default value is |
org.apache.tomcat.jdbc.pool.interceptor.StatementCache
Caches PreparedStatement
and/or CallableStatement
instances on a connection.
The statements are cached per connection.
The count limit is counted globally for all connections that belong to
the same pool. Once the count reaches max
, subsequent
statements are not returned to the cache and are closed immediately.
Attribute | Description |
---|---|
prepared |
(boolean as String) Enable caching of |
callable |
(boolean as String) Enable caching of |
max |
(int as String) Limit on the count of cached statements across
the connection pool.
The default value is |
org.apache.tomcat.jdbc.pool.interceptor.StatementDecoratorInterceptor
See 48392.
Interceptor to wrap statements and result sets in order to prevent access to the
actual connection
using the methods ResultSet.getStatement().getConnection()
and Statement.getConnection()
Attribute | Description |
---|
org.apache.tomcat.jdbc.pool.interceptor.QueryTimeoutInterceptor
Automatically calls java.sql.Statement.setQueryTimeout(seconds)
when a
new statement is created.
The pool itself doesn't timeout the query, it is still up to the JDBC driver to
enforce query timeouts.
Attribute | Description |
---|---|
queryTimeout |
(int as String) The number of seconds to set for the query timeout.
A value less than or equal to zero will disable this feature.
The default value is |
org.apache.tomcat.jdbc.pool.interceptor.SlowQueryReport
Keeps track of query performance and issues log entries when queries exceed a time
threshold of fail.
The log level used is WARN
Attribute | Description |
---|---|
threshold |
(int as String) The number of milliseconds a query has to exceed before
issuing a log alert.
The default value is |
maxQueries |
(int as String) The maximum number of queries to keep track of in order
to preserve memory space.
A value less than or equal to 0 will disable this feature.
The default value is |
logSlow |
(boolean as String) Set to |
logFailed |
(boolean as String) Set to |
org.apache.tomcat.jdbc.pool.interceptor.SlowQueryReportJmx
Extends the SlowQueryReport
and in addition to log entries it issues JMX
notification
for monitoring tools to react to. Inherits all the attributes from its parent class.
This class uses Tomcat's JMX engine so it wont work outside of the Tomcat container.
By default, JMX notifications are sent through the ConnectionPool mbean if it is
enabled.
The SlowQueryReportJmx
can also register an MBean if notifyPool=false
Attribute | Description |
---|---|
notifyPool |
(boolean as String) Set to false if you want JMX notifications to go to
the |
objectName |
(String) Define a valid |
org.apache.tomcat.jdbc.pool.interceptor.ResetAbandonedTimer
The abandoned timer starts when a connection is checked out from the pool.
This means if you have a 30second timeout and run 10x10second queries using the
connection
it will be marked abandoned and potentially reclaimed depending on the abandonWhenPercentageFull
attribute.
Using this interceptor it will reset the checkout timer every time you perform an
operation on the connection or execute a
query successfully.
Attribute | Description |
---|
Code Example
Other examples of Tomcat configuration for JDBC usage can be found in the Tomcat documentation.
Plain Ol' Java
Here is a simple example of how to create and use a data source.
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.Statement;
import org.apache.tomcat.jdbc.pool.DataSource;
import org.apache.tomcat.jdbc.pool.PoolProperties;
public class SimplePOJOExample {
public static void main(String[] args) throws Exception {
PoolProperties p = new PoolProperties();
p.setUrl("jdbc:mysql://localhost:3306/mysql");
p.setDriverClassName("com.mysql.jdbc.Driver");
p.setUsername("root");
p.setPassword("password");
p.setJmxEnabled(true);
p.setTestWhileIdle(false);
p.setTestOnBorrow(true);
p.setValidationQuery("SELECT 1");
p.setTestOnReturn(false);
p.setValidationInterval(30000);
p.setTimeBetweenEvictionRunsMillis(30000);
p.setMaxActive(100);
p.setInitialSize(10);
p.setMaxWait(10000);
p.setRemoveAbandonedTimeout(60);
p.setMinEvictableIdleTimeMillis(30000);
p.setMinIdle(10);
p.setLogAbandoned(true);
p.setRemoveAbandoned(true);
p.setJdbcInterceptors(
"org.apache.tomcat.jdbc.pool.interceptor.ConnectionState;"+
"org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer");
DataSource datasource = new DataSource();
datasource.setPoolProperties(p);
Connection con = null;
try {
con = datasource.getConnection();
Statement st = con.createStatement();
ResultSet rs = st.executeQuery("select * from user");
int cnt = 1;
while (rs.next()) {
System.out.println((cnt++)+". Host:" +rs.getString("Host")+
" User:"+rs.getString("User")+" Password:"+rs.getString("Password"));
}
rs.close();
st.close();
} finally {
if (con!=null) try {con.close();}catch (Exception ignore) {}
}
}
}
As a Resource
And here is an example on how to configure a resource for JNDI lookups
<Resource name="jdbc/TestDB"
auth="Container"
type="javax.sql.DataSource"
factory="org.apache.tomcat.jdbc.pool.DataSourceFactory"
testWhileIdle="true"
testOnBorrow="true"
testOnReturn="false"
validationQuery="SELECT 1"
validationInterval="30000"
timeBetweenEvictionRunsMillis="30000"
maxActive="100"
minIdle="10"
maxWait="10000"
initialSize="10"
removeAbandonedTimeout="60"
removeAbandoned="true"
logAbandoned="true"
minEvictableIdleTimeMillis="30000"
jmxEnabled="true"
jdbcInterceptors="org.apache.tomcat.jdbc.pool.interceptor.ConnectionState;
org.apache.tomcat.jdbc.pool.interceptor.StatementFinalizer"
username="root"
password="password"
driverClassName="com.mysql.jdbc.Driver"
url="jdbc:mysql://localhost:3306/mysql"/>
Asynchronous Connection Retrieval
The Tomcat JDBC connection pool supports asynchronous connection retrieval without
adding additional threads to the
pool library. It does this by adding a method to the data source called Future<Connection>
getConnectionAsync()
.
In order to use the async retrieval, two conditions must be met:
- You must configure the
fairQueue
property to betrue
. - You will have to cast the data source to
org.apache.tomcat.jdbc.pool.DataSource
Connection con = null;
try {
Future<Connection> future = datasource.getConnectionAsync();
while (!future.isDone()) {
System.out.println("Connection is not yet available. Do some background work");
try {
Thread.sleep(100); //simulate work
}catch (InterruptedException x) {
Thread.currentThread().interrupt();
}
}
con = future.get(); //should return instantly
Statement st = con.createStatement();
ResultSet rs = st.executeQuery("select * from user");
Interceptors
Interceptors are a powerful way to enable, disable or modify functionality on a
specific connection or its sub components.
There are many different use cases for when interceptors are useful. By default, and
for performance reasons, the connection pool is stateless.
The only state the pool itself inserts are defaultAutoCommit
, defaultReadOnly
,
defaultTransactionIsolation
, defaultCatalog
if
these are set. These 4 properties are only set upon connection creation. Should
these properties be modified during the usage of the connection,
the pool itself will not reset them.
An interceptor has to extend the
org.apache.tomcat.jdbc.pool.JdbcInterceptor
class. This class is fairly
simple,
You will need to have a no arg constructor
public JdbcInterceptor() {
}
When a connection is borrowed from the pool, the interceptor can initialize or in some other way react to the event by implementing the
public abstract void reset(ConnectionPool parent, PooledConnection con);
method. This method gets called with two parameters, a reference to the connection
pool itself ConnectionPool parent
and a reference to the underlying connection PooledConnection con
.
When a method on the java.sql.Connection
object is invoked, it will
cause the
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
method to get invoked. The Method method
is the actual method invoked,
and Object[] args
are the arguments.
To look at a very simple example, where we demonstrate how to make the invocation to
java.sql.Connection.close()
a noop
if the connection has been closed
if (CLOSE_VAL==method.getName()) {
if (isClosed()) return null; //noop for already closed.
}
return super.invoke(proxy,method,args);
There is an observation being made. It is the comparison of the method name. One way
to do this would be to do
"close".equals(method.getName())
.
Above we see a direct reference comparison between the method name and static
final String
reference.
According to the JVM spec, method names and static final String end up in a shared
constant pool, so the reference comparison should work.
One could of course do this as well:
if (compare(CLOSE_VAL,method)) {
if (isClosed()) return null; //noop for already closed.
}
return super.invoke(proxy,method,args);
The compare(String,Method)
will use the useEquals
flag on
an interceptor and do either reference comparison or
a string value comparison when the useEquals=true
flag is set.
Pool start/stop
When the connection pool is started or closed, you can be notifed. You will only be
notified once per interceptor class
even though it is an instance method. and you will be notified using an interceptor
currently not attached to a pool.
public void poolStarted(ConnectionPool pool) {
}
public void poolClosed(ConnectionPool pool) {
}
When overriding these methods, don't forget to call super if you are extending a
class other than JdbcInterceptor
Configuring interceptors
Interceptors are configured using the jdbcInterceptors
property or the
setJdbcInterceptors
method.
An interceptor can have properties, and would be configured like this
String jdbcInterceptors=
"org.apache.tomcat.jdbc.pool.interceptor.ConnectionState(useEquals=true,fast=yes)"
Interceptor properties
Since interceptors can have properties, you need to be able to read the values of
these properties within your
interceptor. Taking an example like the one above, you can override the setProperties
method.
public void setProperties(Map<String, InterceptorProperty> properties) {
super.setProperties(properties);
final String myprop = "myprop";
InterceptorProperty p1 = properties.get(myprop);
if (p1!=null) {
setMyprop(Long.parseLong(p1.getValue()));
}
}
Getting the actual JDBC connection
Connection pools create wrappers around the actual connection in order to properly
pool them.
We also create interceptors in these wrappers to be able to perform certain
functions.
If there is a need to retrieve the actual connection, one can do so using the javax.sql.PooledConnection
interface.
Connection con = datasource.getConnection();
Connection actual = ((javax.sql.PooledConnection)con).getConnection();
Building
We build the JDBC pool code with 1.6, but it is backwards compatible down to 1.5 for runtime environment. For unit test, we use 1.6 and higher
Other examples of Tomcat configuration for JDBC usage can be found in the Tomcat documentation.
Building from source
Building is pretty simple. The pool has a dependency on tomcat-juli.jar
and in case you want the SlowQueryReportJmx
javac -classpath tomcat-juli.jar \
-d . \
org/apache/tomcat/jdbc/pool/*.java \
org/apache/tomcat/jdbc/pool/interceptor/*.java \
org/apache/tomcat/jdbc/pool/jmx/*.java
A build file can be found in the Tomcat source repository.
As a convenience, a build file is also included where a simple build command will generate all files needed.
ant download (downloads dependencies)
ant build (compiles and generates .jar files)
ant dist (creates a release package)
ant test (runs tests, expects a test database to be setup)
The system is structured for a Maven build, but does generate release artifacts. Just the library itself.