org.smallmind.persistence.sql.AbstractPooledConnection Maven / Gradle / Ivy
/*
* Copyright (c) 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016 David Berkman
*
* This file is part of the SmallMind Code Project.
*
* The SmallMind Code Project is free software, you can redistribute
* it and/or modify it under either, at your discretion...
*
* 1) The terms of GNU Affero General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* ...or...
*
* 2) The terms of the Apache License, Version 2.0.
*
* The SmallMind Code Project is distributed in the hope that it will
* be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License or Apache License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* and the Apache License along with the SmallMind Code Project. If not, see
* or .
*
* Additional permission under the GNU Affero GPL version 3 section 7
* ------------------------------------------------------------------
* If you modify this Program, or any covered work, by linking or
* combining it with other code, such other code is not for that reason
* alone subject to any of the requirements of the GNU Affero GPL
* version 3.
*/
package org.smallmind.persistence.sql;
import java.io.PrintWriter;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import javax.sql.CommonDataSource;
import javax.sql.ConnectionEvent;
import javax.sql.ConnectionEventListener;
import javax.sql.PooledConnection;
import javax.sql.StatementEventListener;
import org.smallmind.nutsnbolts.lang.Existential;
import org.smallmind.nutsnbolts.lang.StaticInitializationError;
public abstract class AbstractPooledConnection implements PooledConnection, InvocationHandler {
private static final Method CLOSE_METHOD;
private final PooledPreparedStatementCache statementCache;
private D dataSource;
private Connection actualConnection;
private Connection proxyConnection;
private ConcurrentLinkedQueue connectionEventListenerQueue;
private ConcurrentLinkedQueue statementEventListenerQueue;
private AtomicBoolean closed = new AtomicBoolean(false);
private long creationMilliseconds;
static {
try {
CLOSE_METHOD = Connection.class.getMethod("close");
} catch (NoSuchMethodException noSuchMethodException) {
throw new StaticInitializationError(noSuchMethodException);
}
}
public AbstractPooledConnection (D dataSource, Connection actualConnection, int maxStatements)
throws SQLException {
this.dataSource = dataSource;
this.actualConnection = actualConnection;
if (maxStatements < 0) {
throw new SQLException("The maximum number of cached statements for this connection must be >= 0");
}
creationMilliseconds = System.currentTimeMillis();
proxyConnection = (Connection)Proxy.newProxyInstance(dataSource.getClass().getClassLoader(), new Class[]{Connection.class, Existential.class}, this);
connectionEventListenerQueue = new ConcurrentLinkedQueue<>();
statementEventListenerQueue = new ConcurrentLinkedQueue<>();
if (maxStatements == 0) {
statementCache = null;
} else {
addStatementEventListener(statementCache = new PooledPreparedStatementCache(maxStatements));
}
}
public abstract ConnectionEvent getConnectionEvent (SQLException sqlException);
public Object invoke (Object proxy, Method method, Object[] args)
throws Throwable {
if (CLOSE_METHOD.equals(method)) {
ConnectionEvent event = getConnectionEvent(null);
for (ConnectionEventListener listener : connectionEventListenerQueue) {
listener.connectionClosed(event);
}
return null;
} else {
try {
if ((statementCache != null) && PreparedStatement.class.isAssignableFrom(method.getReturnType())) {
PreparedStatement preparedStatement;
synchronized (statementCache) {
if ((preparedStatement = statementCache.getPreparedStatement(args)) == null) {
preparedStatement = statementCache.cachePreparedStatement(args, new PooledPreparedStatement(this, (PreparedStatement)method.invoke(actualConnection, args)));
}
}
return preparedStatement;
} else {
return method.invoke(actualConnection, args);
}
} catch (Throwable throwable) {
Throwable closestCause;
closestCause = ((throwable instanceof InvocationTargetException) && (throwable.getCause() != null)) ? throwable.getCause() : throwable;
if (closestCause instanceof SQLException) {
ConnectionEvent event = getConnectionEvent((SQLException)closestCause);
for (ConnectionEventListener listener : connectionEventListenerQueue) {
listener.connectionErrorOccurred(event);
}
}
throw new PooledConnectionException(closestCause, "Connection encountered an exception after operation for %d milliseconds", System.currentTimeMillis() - creationMilliseconds);
}
}
}
public PrintWriter getLogWriter ()
throws SQLException {
return dataSource.getLogWriter();
}
public Connection getConnection ()
throws SQLException {
return proxyConnection;
}
public void close ()
throws SQLException {
if (closed.compareAndSet(false, true)) {
try {
if (statementCache != null) {
try {
statementCache.close();
} finally {
removeStatementEventListener(statementCache);
}
}
} finally {
actualConnection.close();
}
}
}
public void finalize ()
throws SQLException {
try {
close();
} catch (SQLException sqlException) {
PrintWriter logWriter;
if ((logWriter = getLogWriter()) != null) {
sqlException.printStackTrace(logWriter);
}
}
}
public void addConnectionEventListener (ConnectionEventListener listener) {
connectionEventListenerQueue.add(listener);
}
public void removeConnectionEventListener (ConnectionEventListener listener) {
connectionEventListenerQueue.remove(listener);
}
protected Iterable getStatementEventListeners () {
return statementEventListenerQueue;
}
public void addStatementEventListener (StatementEventListener listener) {
statementEventListenerQueue.add(listener);
}
public void removeStatementEventListener (StatementEventListener listener) {
statementEventListenerQueue.remove(listener);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy