org.tentackle.dbms.ManagedConnection Maven / Gradle / Ivy
Show all versions of tentackle-database Show documentation
/*
* Tentackle - https://tentackle.org
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package org.tentackle.dbms;
import org.tentackle.common.Timestamp;
import org.tentackle.dbms.rmi.RemoteDbSessionImpl;
import org.tentackle.io.ReconnectedException;
import org.tentackle.log.Logger;
import org.tentackle.log.Logger.Level;
import org.tentackle.log.MappedDiagnosticContext;
import org.tentackle.session.PersistenceException;
import org.tentackle.session.SessionClosedException;
import org.tentackle.session.TransactionIsolation;
import org.tentackle.session.TransactionWritability;
import org.tentackle.sql.Backend;
import java.lang.ref.Cleaner;
import java.lang.ref.Cleaner.Cleanable;
import java.lang.ref.WeakReference;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.SQLWarning;
import java.sql.Savepoint;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.Collection;
import java.util.ConcurrentModificationException;
import java.util.Date;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong;
import java.util.regex.Pattern;
/**
* A jdbc connection managed by the ConnectionManager.
*
* The connection provides some additional features
* such as a prepared statement cache
* and translates SQLExceptions to PersistenceExceptions.
* The ConnectionManager is responsible to attach and detach
* the connection to a session.
*
* @author harald
*/
public class ManagedConnection {
/**
* Minimum milliseconds a connection remains attached before being logged.
* 0 to disable, -1 to log every detach.
*
* For analysis and debugging purposes only.
*/
private static long logMinAttachMillis;
/**
* Flag to turn on statement statistics. (default is turned off)
*/
private static boolean collectStatistics;
/**
* Optional periodical log of statistics in milliseconds. (default is off)
*
* If turned on, the statistics will be logged and cleared every logStatisticsMillis milliseconds.
*/
private static long logStatisticsMillis;
/**
* Optional filter to log only statements for certain MappedDiagnosticContexts.
* If set, statements are only collected for logging if the MDC is valid and the pattern matches the toString-value
* of the MDC.
*/
private static Pattern mdcFilter;
/**
* Optionally enable statement checks for resultSetType and resultSetConcurrency.
*/
private static boolean checkStatementResultSetParameters;
/**
* Logging level to log nested selects.
* When enabled, selects running in parallel on the same connection will be logged.
* This is usually means, that from within a loop processing a result set, a method is invoked,
* that in turn triggers another select.
*/
private static Level logLevelForParallelOpenResultSets;
/**
* Optional limit for the number of statements per transaction.
* Useful to detect wild running transactions eating up all memory and blowing up the backend.
* Default is 0 (disabled).
*/
private static int maxStatementsPerTransaction;
/**
* Gets the minimum milliseconds a connection remains attached before being logged.
*
* @return the minimum milliseconds
*/
public static long getLogMinAttachMillis() {
return logMinAttachMillis;
}
/**
* Sets the minimum milliseconds a connection remains attached before being logged.
*
* @param millis milliseconds, 0 to disable, -1 to log every detach
*/
public static void setLogMinAttachMillis(long millis) {
logMinAttachMillis = millis;
}
/**
* Returns whether to collect statement statistics.
*
* @return true if statistics enabled
*/
public static boolean isCollectingStatistics() {
return collectStatistics;
}
/**
* Sets a flag whether to collect statement statistics.
*
* @param collect true to count invocations and execution times
*/
public static void setCollectingStatistics(boolean collect) {
collectStatistics = collect;
}
/**
* Gets the optional periodical log of statistics in milliseconds.
*
* @return the period in milliseconds, 0 if disabled (default)
*/
public static long getLogStatisticsMillis() {
return logStatisticsMillis;
}
/**
* Sets the optional periodical log of statistics in milliseconds.
* If turned on, the statistics will be logged and cleared every logStatisticsMillis milliseconds.
*
* @param millis the period in milliseconds, 0 to disable
*/
public static void setLogStatisticsMillis(long millis) {
ManagedConnection.logStatisticsMillis = millis;
}
/**
* Gets the MDC filter.
*
* @return the filter, null if none
*/
public static Pattern getMdcFilter() {
return mdcFilter;
}
/**
* Sets Optional filter to log only statements for certain MappedDiagnosticContexts.
* If set, statements are only collected for logging if the MDC is valid and the pattern matches the toString-value
* of the MDC.
*
* @param filter the filter, null to clear
*/
public static void setMdcFilter(Pattern filter) {
mdcFilter = filter;
}
/**
* Returns whether to check that requested resulttype and concurrency matches the prepared statement reused.
*
* @return true if check is enabled
*/
public static boolean isCheckingStatementResultSetParameters() {
return checkStatementResultSetParameters;
}
/**
* Sets whether to check that requested resulttype and concurrency matches the prepared statement reused.
*
* @param check true to enable check
*/
public static void setCheckingStatementResultSetParameters(boolean check) {
checkStatementResultSetParameters = check;
}
/**
* Gets the logging level to log nested selects.
*
* @return the level, null if disabled
*/
public static Level getLogLevelForParallelOpenResultSets() {
return logLevelForParallelOpenResultSets;
}
/**
* Sets the logging level to log nested selects.
* If enabled, selects running in parallel on the same connection will be logged.
* This is usually means, that from within a loop processing a result set, a method is invoked,
* that in turn triggers another select.
*
* @param level the logging level, null to disable
*/
public static void setLogLevelForParallelOpenResultSets(Level level) {
logLevelForParallelOpenResultSets = level;
}
/**
* Sets the limit for the number of statements per transaction.
* Useful to detect wild running transactions eating up all memory and blowing up the backend.
* Default is 0 (disabled).
*
* @param max the max. number of statements within a transaction.
*/
public static void setMaxStatementsPerTransaction(int max) {
maxStatementsPerTransaction = max;
}
/**
* Gets the statement limit per transaction.
*
* @return number of statements, 0 if unlimited (default)
*/
public static int getMaxStatementsPerTransaction() {
return maxStatementsPerTransaction;
}
private static final Logger LOGGER = Logger.get(ManagedConnection.class);
private static final AtomicLong INSTANCE_COUNTER = new AtomicLong();
private static final Cleaner CLEANER = Cleaner.create(); // instead of deprecated finalize()
/**
* Holds the resources to be cleaned up.
* Notice that ManagedConnections are referenced by their ConnectionManager.
* As a consequence, they will be cleaned up when the ConnectionManager is cleaned up.
*/
private static class Resources implements Runnable {
private Connection connection; // the wrapped connection, null if closed
private String name; // the name of the managed connection
private boolean dead; // connection is dead (comlink error detected)
private ManagedConnection me; // != null if programmatic cleanup
Resources(Connection connection) {
this.connection = connection;
}
@Override
public void run() {
try {
Connection c = connection;
if (c != null) {
if (me == null) {
// application lost the reference and did not close it before
LOGGER.warning("closing unreferenced connection {0}", name);
}
try {
c.close();
}
catch (SQLException | RuntimeException sx) {
if (!dead) {
LOGGER.severe("low level close failed for " + name + " -> marked closed", sx);
}
}
finally {
connection = null;
}
}
}
finally {
me = null;
}
}
}
private final Resources resources; // resources to be cleaned up
private final Cleanable cleanable; // to clean up the connection
private final ConnectionManager manager; // the manager that created this connection
private final Backend backend; // the backend
private final String backendId; // a unique ID to identify the connection at the backend
private final long instanceNumber; // unique instance number (for logging only)
private final long establishedSince; // connection established establishedSince... (epochal [ms])
private final int defaultTxIsolation; // the initial/default transaction isolation level
private final boolean defaultReadOnly; // the initial/default read-only mode
private int txIsolation; // the current transaction isolation level
private boolean readOnly; // the current read-only mode
private Db db; // currently attached Db, null = free connection
private long expireAt; // connection shutdown at (epochal [ms]), 0 = forever
private long attachedSince; // attached since when...
private long detachedSince; // detached since when...
private long lastVerified; // last connection verification
private int index; // connection index given by connection manager, -1 not used
private int attachCount; // 0 = not attached, else number of times db is attached
private int maxCountForClearWarnings; // trigger when to clearWarning() on a connection (0 = disabled)
private int counterForClearWarnings; // current counter
private StatementHistory currentStatement; // statement history while not in transaction
private long connectionInactivityTimeoutMs; // optional database connection inactivity timeout
private boolean connectionKeepAliveEnabled; // optional periodical dummy select to prevent premature close of idle connection
private final List statementHistory; // statement history while in transaction
private final Map preparedStatements; // all prepared statements for this connection (stmtId:statement)
private final Set> openResultSets; // pending open result sets (thread safe bec. of WeakRefs)
private final Set> runningStatements; // running statements (thread safe bec. of WeakRefs)
/**
* Creates a managed connection.
*
* @param manager the connection manager
* @param backend the backend
* @param connection the low level JDBC connection
*/
public ManagedConnection(ConnectionManager manager, Backend backend, Connection connection) {
this.manager = Objects.requireNonNull(manager, "manager");
this.backend = Objects.requireNonNull(backend, "backend");
backendId = backend.getBackendId(Objects.requireNonNull(connection, "connection"));
instanceNumber = INSTANCE_COUNTER.incrementAndGet();
establishedSince = System.currentTimeMillis();
detachedSince = establishedSince;
index = -1;
statementHistory = new ArrayList<>();
preparedStatements = new ConcurrentHashMap<>();
// use a thread-safe set due to concurrent cleanup of weak references in getOpenResultSets and cancelRunningStatements
openResultSets = ConcurrentHashMap.newKeySet();
runningStatements = ConcurrentHashMap.newKeySet();
resources = new Resources(connection);
resources.name = toString(); // cannot be a constructor arg because of chicken egg
cleanable = CLEANER.register(this, resources);
defaultTxIsolation = txIsolation = getTransactionIsolation();
defaultReadOnly = readOnly = isReadOnly();
ManagedConnectionMonitor.getInstance().registerManagedConnection(this);
}
/**
* Gets the connection manager
*
* @return the manager
*/
public ConnectionManager getManager() {
return manager;
}
/**
* Gets the backend.
*
* @return the backend
*/
public Backend getBackend() {
return backend;
}
/**
* Gets the low level connection.
*
* @return the physical connection, never null
* @throws PersistenceException if connection is closed
*/
public Connection getConnection() {
Connection c = resources.connection;
if (c == null) {
throw new PersistenceException("connection is closed");
}
return c;
}
/**
* Gets the optional connection inactivity timeout.
*
* @return the timeout in [ms]
*/
public long getConnectionInactivityTimeoutMs() {
return connectionInactivityTimeoutMs;
}
/**
* Sets the optional connection inactivity timeout.
*
* @param connectionInactivityTimeoutMs the timeout in [ms]
*/
public void setConnectionInactivityTimeoutMs(long connectionInactivityTimeoutMs) {
this.connectionInactivityTimeoutMs = connectionInactivityTimeoutMs;
}
/**
* Returns whether dummy selects are sent periodically to the backend.
*
* @return true if idle connections kept alive
*/
public boolean isConnectionKeepAliveEnabled() {
return connectionKeepAliveEnabled;
}
/**
* Sets the optional flag to send dummy selects periodically to the backend.
*
* @param connectionKeepAliveEnabled true to keep idle connections alive
*/
public void setConnectionKeepAliveEnabled(boolean connectionKeepAliveEnabled) {
this.connectionKeepAliveEnabled = connectionKeepAliveEnabled;
}
/**
* Gets the epochal time in ms when this connection was verified.
*
* @return the last verification, 0 if never been verified at all
*/
public long getLastVerified() {
return lastVerified;
}
/**
* Gets the epochal time when this connection was established.
*
* @return the time establishedSince in ms
*/
public long getEstablishedSince() {
return establishedSince;
}
/**
* Sets the epochal time when this connection should be closed, if unused.
*
* @param expireAt the time in [ms], 0 = forever
*/
public void setExpireAt(long expireAt) {
this.expireAt = expireAt;
}
/**
* Gets the epochal time when this connection should be closed, if unused.
*
* @return the time in [ms], 0 = forever
*/
public long getExpireAt() {
return expireAt;
}
/**
* Gets the epochal time when this connection was attached.
*
* @return the epochal time of last attach, 0 if not attached
*/
public long getAttachedSince() {
return attachedSince;
}
/**
* Gets the epochal time when this connection was detached.
*
* Note that newly created connections get their detach time initialized
* from the current system time.
*
* @return the epochal time of last detach, 0 if attached
*/
public long getDetachedSince() {
return detachedSince;
}
/**
* Sets the connection index.
* Connection managers use that to manage connection lists.
*
* @param index the connection index
*/
public void setIndex(int index) {
this.index = index;
}
/**
* Gets the connection index.
*
* @return the connection index.
*/
public int getIndex() {
return index;
}
/**
* Marks a connection being dead.
*
* Marking connections dead allows connection managers like
* {@link MpxConnectionManager} to re-open connections before
* being attached next time. Notice that not all connection managers
* honour the dead-flag (makes only sense in servers, anyway).
*
* @param dead true if marked dead, false not dead
*/
public void setDead(boolean dead) {
if (dead && !resources.dead) {
LOGGER.severe("managed connection {0}: *** MARKED DEAD ***", this);
}
resources.dead = dead;
}
/**
* Returns whether connection is marked dead
* @return true if dead
*/
public boolean isDead() {
return resources.dead;
}
/**
* Returns whether timeout is enabled and connection wasn't use for a long time and needs verification.
*
* @return return true if {@link #verifyConnection()} should be invoked before use
*/
public boolean isConnectionVerificationNecessary() {
return isConnectionVerificationNecessary(System.currentTimeMillis());
}
/**
* Returns whether timeout is enabled and connection wasn't use for a long time and needs verification.
*
* @param currentTimeMillis the current epochal time in milliseconds
* @return return true if {@link #verifyConnection()} should be invoked before use
*/
public boolean isConnectionVerificationNecessary(long currentTimeMillis) {
return connectionInactivityTimeoutMs > 0 && detachedSince > 0 &&
currentTimeMillis - detachedSince >= connectionInactivityTimeoutMs;
}
/**
* Checks whether connection is still valid.
* Usually implemented by the backend via a "SELECT 1" query.
* If the check fails the connection is marked dead.
*
* @return true if connection still valid, false if invalid
* @see Backend#getDummySelect()
*/
public boolean verifyConnection() {
try (Statement stmt = getConnection().createStatement()) {
// Can be invoked at any time, even if another statement is currently running
// on this connection (submitted from another thread) since JDBC-drivers are thread-safe these days.
// However, the invocation may be blocked until the other statement finishes.
stmt.executeQuery(getBackend().getDummySelect());
lastVerified = System.currentTimeMillis();
return true;
} // stmt.close closes the result set as well
catch (RuntimeException | SQLException ex) {
setDead(true);
return false;
}
}
/**
* Attaches a connection to a session.
* Connections must be attached before it can be used by statements
* or starting a tx.
* The method must only be invoked by a connection manager!
*
* @param db the logical Db to attach
*/
public void attachSession(Db db) {
Objects.requireNonNull(db, "db");
if (this.db != null) {
if (this.db == db) {
if (db.getConnection() != this) {
throw new PersistenceException(db, "db lost current connection " + this + ", count=" + attachCount);
}
attachCount++;
}
else {
throw new PersistenceException(db, "connection " + this + " already attached to " + this.db);
}
}
else {
if (attachCount != 0) {
throw new PersistenceException(db, "attach count of unattached connection " + this + " is not 0, but " + attachCount);
}
this.db = db;
db.setConnection(this);
attachCount = 1;
attachedSince = System.currentTimeMillis();
detachedSince = 0;
}
LOGGER.finer("{0} attached to {1}, count={2}", db, this, attachCount);
}
/**
* Checks whether a session is attached to this connection.
*
* @return true if attached, else false.
*/
public boolean isAttached() {
return db != null;
}
/**
* Returns whether a transaction is running.
*
* @return true if connection is attached to a session in transaction
*/
public boolean isTxRunning() {
return isAttached() && db.isTxRunning();
}
/**
* Returns whether connection has expired.
* Expired connections should be closed and reopened.
*
* @return true if expired
*/
public boolean isExpired() {
return expireAt > 0 && expireAt < detachedSince;
}
/**
* Detaches a session from a connection.
* Connections must be detached before they can be used by another Db.
* The method must only be invoked by a connection manager!
*
* @param db the db to detach
*/
public void detachSession(Db db) {
Objects.requireNonNull(db, "db");
if (this.db != db) {
throw new PersistenceException(db, "connection " + this + " not attached to " + db + " (instead attached to " + this.db + ")");
}
LOGGER.finer("{0} detached from {1}, count={2}", db, this, attachCount);
attachCount--;
if (attachCount == 0) {
detachedSince = System.currentTimeMillis();
if (logMinAttachMillis != 0 && detachedSince - attachedSince > logMinAttachMillis) {
if (isMDCValid()) {
LOGGER.warning("attached for {0}ms: {1}", () -> detachedSince - attachedSince, this::toDiagnosticString);
}
}
else if (isMDCValid()) {
LOGGER.fine(this::toDiagnosticString);
}
this.db = null;
db.setConnection(null);
attachedSince = 0;
if (collectStatistics) {
if (logStatisticsMillis > 0 && detachedSince - StatementStatistics.getSince() > logStatisticsMillis) {
// log and clear statistics
StatementStatistics.log(Logger.Level.INFO, true);
DbTransactionFactory.getInstance().logStatistics(Logger.Level.INFO, true);
}
// collect all statements
if (isMDCValid()) {
for (StatementHistory history: statementHistory) {
StatementStatistics.collect(history);
}
if (currentStatement != null) {
StatementStatistics.collect(currentStatement);
}
}
}
statementHistory.clear();
currentStatement = null;
restoreTransactionDefaults();
}
else {
if (db.getConnection() != this) {
throw new PersistenceException(db, "db lost current connection " + this + ", count=" + attachCount++);
}
if (attachCount < 0) {
throw new PersistenceException(db, "connection " + this + " detached too often, attachcount = " + attachCount++);
}
}
}
/**
* Gets tha attached session.
*
* @return the session, null if not attached
*/
public Db getSession() {
return db;
}
/**
* Gets the unique instance number.
*
* @return the instance number
*/
public long getInstanceNumber() {
return instanceNumber;
}
/**
* Gets the backend id.
*
* @return the unique id
*/
public String getBackendId() {
return backendId;
}
/**
* Gets the connection name.
*
* @return the unique name
*/
public String getName() {
StringBuilder buf = new StringBuilder();
buf.append(instanceNumber);
if (backendId != null) {
buf.append('/').append(backendId);
}
return buf.toString();
}
@Override
public String toString() {
StringBuilder buf = new StringBuilder();
Connection con = resources.connection; // local variable because could be closed between test and use
if (con == null) {
buf.append("");
}
else {
buf.append(con);
}
buf.append('[').append(getName());
if (index >= 0) {
buf.append('/').append(index);
}
buf.append(']');
if (attachCount > 0) {
buf.append(", attached ").append(attachCount).append(" since ").append(new Timestamp(attachedSince));
}
if (resources.dead) {
buf.append(", *DEAD*");
}
return buf.toString();
}
/**
* Checks whether the mapped diagnostic context is valid for statistics logging.
*
* @return true if loggable
* @see #mdcFilter
*/
protected boolean isMDCValid() {
if (mdcFilter == null) {
return true;
}
MappedDiagnosticContext mdc = LOGGER.getMappedDiagnosticContext();
return mdc != null && mdc.matchesPattern(mdcFilter);
}
/**
* Logs the execution start of a statement during a transaction.
* The execution start is the current epochal time.
* The execution end must be set via {@link StatementHistory#end()}
*
* @param statement the statement
* @param sql optional SQL string if statement does not provide it
* @return the history with an execution start of now and execution end not set, never null
*/
StatementHistory logStatementHistory(StatementWrapper statement, String sql) {
StatementHistory history = new StatementHistory(statement, sql);
if (isTxRunning()) {
statementHistory.add(history);
if (maxStatementsPerTransaction != 0 && statementHistory.size() > maxStatementsPerTransaction) {
throw new PersistenceException(db, "transaction limit of statements exceeded (" + maxStatementsPerTransaction + ")");
}
}
else {
currentStatement = history;
}
return history;
}
/**
* Gets the statement history.
*
* @return the statements executed during last attachment
*/
public List getStatementHistory() {
return statementHistory;
}
/**
* Adds a pending result set.
*
* @param rs the result set
*/
public void addResultSet(ResultSetWrapper rs) {
openResultSets.add(new WeakReference<>(rs));
if (logLevelForParallelOpenResultSets != null && openResultSets.size() > 1 && LOGGER.isLoggable(logLevelForParallelOpenResultSets)) {
Collection parallelResultStatements = getOpenResultSets();
if (!parallelResultStatements.isEmpty()) {
int count = 0;
StringBuilder buf = new StringBuilder();
buf.append("parallel queries running on ").append(this).append(":");
for (ResultSetWrapper prs: parallelResultStatements) {
StatementWrapper stmt = prs.getStatement();
if (stmt != null && !stmt.isParallelOk()) {
buf.append('\n').append(prs);
count++;
}
}
if (count > 1) {
LOGGER.log(logLevelForParallelOpenResultSets, buf.toString(), null);
}
}
}
}
/**
* Removes a pending result set.
*
* @param rs the result set
*/
public void removeResultSet(ResultSetWrapper rs) {
for (Iterator> iter = openResultSets.iterator(); iter.hasNext(); ) {
WeakReference ref = iter.next();
ResultSetWrapper refRs = ref.get();
// if closed or unreferenced or the db to remove
if (refRs == null || refRs == rs) { // == is okay here
iter.remove();
}
}
}
/**
* Gets all open result sets for this connection.
*
* @return the result sets, never null
*/
public Collection getOpenResultSets() {
List rsList = new ArrayList<>();
for (Iterator> iter = openResultSets.iterator(); iter.hasNext(); ) {
WeakReference ref = iter.next();
ResultSetWrapper refRs = ref.get();
if (refRs != null) {
rsList.add(refRs);
}
else {
iter.remove();
}
}
return rsList;
}
/**
* Creates a diagnostic string.
*
* @return the connection's status info
*/
public String toDiagnosticString() {
StringBuilder buf = new StringBuilder(toString());
buf.append(": ")
.append(manager)
.append('[')
.append(index)
.append("] valid ")
.append(new Date(establishedSince))
.append(" - ");
if (expireAt != 0) {
buf.append(new Date(expireAt));
}
else {
buf.append(" ");
}
if (readOnly != defaultReadOnly) {
buf.append(", ").append(TransactionWritability.valueOf(!readOnly));
}
if (txIsolation != defaultTxIsolation) {
buf.append(", ").append(TransactionIsolation.valueOf(txIsolation));
}
Db lDb = db; // may be closed in the meantime...
if (lDb != null) {
buf.append("\n -> ").append(lDb.getName());
if (lDb.getTxName() != null) {
buf.append('/').append(lDb.getTxName());
}
Thread ownerThread = lDb.getOwnerThread();
if (ownerThread != null) {
buf.append(", ").append(lDb.getOwnerThread());
}
for (RemoteDbSessionImpl session: RemoteDbSessionImpl.getOpenSessions()) {
if (session.getSession() == lDb) {
// this is my session!
buf.append(", ").append(session);
}
}
}
try {
if (!statementHistory.isEmpty() || currentStatement != null) {
buf.append("\n\n Statements:");
try {
buf.append(logStatementHistory(statementHistory));
}
catch (ConcurrentModificationException cmx) {
buf.append(" (history may be incomplete because connection is still in use by other threads)");
buf.append(logStatementHistory(new ArrayList<>(statementHistory))); // try on copy
}
StatementHistory stmt = currentStatement;
if (stmt != null) {
buf.append("\n ");
buf.append(stmt);
}
buf.append("\n");
}
}
catch (RuntimeException rex) {
LOGGER.warning("cannot dump statement history", rex);
}
try {
// dump all open result sets (threadsafe)
Collection rsList = getOpenResultSets();
if (!rsList.isEmpty()) {
buf.append("\n\n Resultsets:");
for (ResultSetWrapper rs: rsList) {
buf.append("\n ");
buf.append(rs);
}
buf.append("\n");
}
}
catch (RuntimeException rex) {
LOGGER.warning("cannot dump open result sets", rex);
}
return buf.toString();
}
private StringBuilder logStatementHistory(Collection statementHistory) {
StringBuilder buf = new StringBuilder();
for (StatementHistory stmt : statementHistory) {
buf.append("\n ").append(stmt);
}
return buf;
}
/**
* asserts that a connection is attached
*/
private void assertAttached() {
if (db == null) {
throw new PersistenceException("connection " + this + " not attached to any Db");
}
}
/**
* Gets the current transaction isolation level.
*
* @return the isolation level
* @see Connection#getTransactionIsolation()
*/
public int getTransactionIsolation() {
try {
return getConnection().getTransactionIsolation();
}
catch (SQLException sx) {
throw createFromSqlException("getting transaction isolation level failed", sx);
}
}
/**
* Sets the transaction isolation level.
*
* @param txIsolation the isolation level
* @see Connection#setTransactionIsolation(int)
*/
public void setTransactionIsolation(int txIsolation) {
try {
getConnection().setTransactionIsolation(txIsolation);
this.txIsolation = txIsolation;
}
catch (SQLException sx) {
throw createFromSqlException("setting transaction isolation level to " + txIsolation + " failed", sx);
}
}
/**
* Gets the read-only mode.
*
* @return true if read-only
* @see Connection#isReadOnly()
*/
public boolean isReadOnly() {
try {
return getConnection().isReadOnly();
}
catch (SQLException sx) {
throw createFromSqlException("getting read-only mode failed", sx);
}
}
/**
* Sets the read-only mode.
*
* @param readOnly true if read-only
* @see Connection#setReadOnly(boolean)
*/
public void setReadOnly(boolean readOnly) {
try {
getConnection().setReadOnly(readOnly);
this.readOnly = readOnly;
}
catch (SQLException sx) {
throw createFromSqlException("setting read-only mode to " + readOnly + " failed", sx);
}
}
/**
* Sets the autocommit feature.
*
* Restores the default transaction isolation level and read-only mode, if set to true and isolation or read-only mode was changed.
*
* @param autoCommit true to enable autocommit, false to disable.
*/
public void setAutoCommit(boolean autoCommit) {
assertAttached();
try {
getConnection().setAutoCommit(autoCommit);
if (autoCommit) {
restoreTransactionDefaults();
}
}
catch (SQLException ex) {
throw createFromSqlException("setting autocommit to " + autoCommit + " failed", ex);
}
}
/**
* Gets the autocommit value.
*
* @return the autocommit value.
*/
public boolean getAutoCommit() {
try {
return getConnection().getAutoCommit();
}
catch (SQLException ex) {
throw createFromSqlException("getting autocommit failed", ex);
}
}
/**
* Performs a commit.
*/
public void commit() {
assertAttached();
try {
getConnection().commit();
}
catch (SQLException ex) {
throw createFromSqlException("commit failed", ex);
}
}
/**
* Performs a rollback.
*
* @param withLog true if log via INFO
*/
public void rollback(boolean withLog) {
assertAttached();
try {
if (withLog) {
LOGGER.info(this::toDiagnosticString);
}
getConnection().rollback();
// unmark all statements marked ready
unmarkPreparedStatements();
}
catch (SQLException ex) {
throw createFromSqlException("rollback failed: " + toDiagnosticString(), ex);
}
}
/**
* Creates an unnamed savepoint in the current transaction.
*
* @return the savepoint
*/
public Savepoint setSavepoint() {
assertAttached();
try {
return getConnection().setSavepoint();
}
catch (SQLException ex) {
throw createFromSqlException("setting unnamed savepoint failed: " + toDiagnosticString(), ex);
}
}
/**
* Creates a savepoint with the given name in the current transaction.
*
* @param name the savepoint name
*
* @return the savepoint
*/
public Savepoint setSavepoint(String name) {
assertAttached();
try {
return getConnection().setSavepoint(name);
}
catch (SQLException ex) {
throw createFromSqlException("setting savepoint '" + name + "' failed: " + toDiagnosticString(), ex);
}
}
/**
* Undoes all changes made after the given Savepoint
object was set.
*
* @param savepoint the savepoint
*/
public void rollback(Savepoint savepoint) {
assertAttached();
try {
getConnection().rollback(savepoint);
}
catch (SQLException ex) {
throw createFromSqlException("rollback of savepoint '" + savepoint + "' failed: " + toDiagnosticString(), ex);
}
}
/**
* Removes the specified Savepoint
and subsequent Savepoint
objects from the current
* transaction.
*
* @param savepoint the savepoint to be removed
*/
public void releaseSavepoint(Savepoint savepoint) {
assertAttached();
if (db.getBackend().isReleaseSavepointSupported()) {
try {
getConnection().releaseSavepoint(savepoint);
}
catch (SQLException ex) {
throw createFromSqlException("release of savepoint '" + savepoint + "' failed: " + toDiagnosticString(), ex);
}
}
}
/**
* Remembers a statement as running.
*
* @param statement the statement
*/
public void addRunningStatement(StatementWrapper statement) {
runningStatements.add(new WeakReference<>(statement));
}
/**
* Removes a statement from the running statements.
*
* @param statement the statement
*/
public void removeRunningStatement(StatementWrapper statement) {
// usually there is no more than one running statement at a time, so the loop is fast
for (Iterator> iter = runningStatements.iterator(); iter.hasNext(); ) {
WeakReference ref = iter.next();
StatementWrapper stmt = ref.get();
if (stmt == null || stmt == statement) {
iter.remove();
}
}
}
/**
* Cancels the(all) currently running statement(s).
* May be invoked from any thread.
*/
public void cancelRunningStatements() {
for (Iterator> iter = runningStatements.iterator(); iter.hasNext(); ) {
WeakReference ref = iter.next();
StatementWrapper stmt = ref.get();
if (stmt != null && stmt.isRunning()) {
try {
stmt.cancel();
}
catch (RuntimeException ex) {
// just log
LOGGER.warning("canceling statement failed: " + stmt, ex);
}
}
iter.remove();
}
}
/**
* Unmark all prepared statements that are marked ready.
*/
private void unmarkPreparedStatements() {
// unmark all statements marked ready
for (PreparedStatementWrapper stmt: preparedStatements.values()) {
if (stmt != null && stmt.isMarkedReady()) {
try {
stmt.consume();
}
catch (RuntimeException ex) {
// just log (if not cancelled)
if (!stmt.isCancelled()) {
LOGGER.warning("unmarking statement failed: " + stmt, ex);
}
}
}
}
}
/**
* Reads all warnings, logs and clears them.
*/
public void logAndClearWarnings() {
try {
// log connection warning
SQLWarning warning = getConnection().getWarnings();
while (warning != null) {
LOGGER.warning(warning.getMessage());
warning = warning.getNextWarning();
}
getConnection().clearWarnings(); // release memory used by warnings
// log prepared statement warnings
for (PreparedStatementWrapper stmt: preparedStatements.values()) {
if (!stmt.isClosed()) { // if not closed
warning = stmt.getStatement().getWarnings();
while (warning != null) {
LOGGER.warning(warning.getMessage());
warning = warning.getNextWarning();
}
stmt.getStatement().clearWarnings();
}
}
// as this statement is called when there are no more running statements
// we ignore the unprepared running statements.
}
catch (SQLException e) {
if (!resources.dead) {
throw createFromSqlException("reading warnings failed", e);
}
}
}
/**
* Sets the countForClearWarnings trigger.
*
* @param max the maxcount, 0 = app must eat the warnings
*/
public void setMaxCountForClearWarnings(int max) {
maxCountForClearWarnings = max;
}
/**
* Gets the current setting for clearWarnings() trigger.
*
* @return the maxcount, 0 = app must eat the warnings
*/
public int getMaxCountForClearWarnings() {
return maxCountForClearWarnings;
}
/**
* Increments a counter and empties the warnings on the connection
* and all prepared statements if a trigger value is reached.
* This is necessary for apps that don't
* clearWarnings() on their own, otherwise filling up memory might occur.
*/
public void countForClearWarnings() {
if (maxCountForClearWarnings > 0) {
counterForClearWarnings++;
if (counterForClearWarnings >= maxCountForClearWarnings) {
logAndClearWarnings();
counterForClearWarnings = 0;
}
}
}
/**
* Forces the connection to be detached from the session.
* The method is used from connection managers only, hence package scope.
*/
void forceDetached() {
if (isAttached()) {
// database is still attached
LOGGER.warning("forcing detach of connection {0}, attached to {1}", this, db);
// close pending statements
closePreparedStatements(true);
if (!isClosed()) {
try {
if (!getAutoCommit()) {
// in transaction!!! rollback
LOGGER.severe("rolling back pending transaction for {0}", db);
rollback(true);
}
}
catch (PersistenceException pex) {
if (!resources.dead) {
LOGGER.severe("connection is broken", pex);
}
closeImpl();
}
}
if (db != null) {
// if not already detached via closePreparedStatements(true)
db.setConnection(null);
db = null;
}
attachCount = 0;
statementHistory.clear();
currentStatement = null;
openResultSets.clear();
restoreTransactionDefaults();
}
// else already detached
}
/**
* Closes the connection.
* Closing an already closed connection is allowed.
*/
public void close() {
if (!isClosed()) {
try {
// rollback before close to prevent an implicit commit (JDBC implementation specific behaviour)
if (!getConnection().getAutoCommit()) {
LOGGER.warning("closing connection while in transaction:\n{0}", this::toDiagnosticString);
getConnection().rollback();
}
}
catch (SQLException ex) {
if (!resources.dead) {
LOGGER.warning("low level rollback failed: {0}", ex.getMessage());
}
}
logAndClearWarnings();
forceDetached();
closePreparedStatements(false);
closeImpl();
}
}
/**
* Closes the connection.
*/
private void closeImpl() {
resources.me = this;
cleanable.clean();
}
/**
* Gets the connection's closed state.
*
* @return true if connection is closed
*/
public boolean isClosed() {
return resources.connection == null;
}
/**
* Closes prepared statements.
*
* @param onlyMarkedReady true if close only pending statements, false if all
*/
public void closePreparedStatements(boolean onlyMarkedReady) {
// close all statements
for (Iterator iter = preparedStatements.values().iterator(); iter.hasNext(); ) {
PreparedStatementWrapper stmt = iter.next();
if (!stmt.isClosed() && // if not already closed
(!onlyMarkedReady || stmt.isMarkedReady())) { // or all or only pending
try {
stmt.close();
}
catch (RuntimeException ex) {
// just log, we need to close them all
if (!resources.dead) {
LOGGER.warning("closing statement failed: " + stmt, ex);
}
}
}
if (onlyMarkedReady) {
iter.remove();
}
}
if (!onlyMarkedReady) {
preparedStatements.clear();
}
}
/**
* Removes a closed prepared statement.
*
* @param st the statement
*/
public void removePreparedStatement(PreparedStatementWrapper st) {
if (!st.isClosed()) {
throw new PersistenceException(db, "statement still open: " + st);
}
StatementKey key = st.getStatementKey();
if (key != null) {
// if not a one-shot
preparedStatements.remove(st.getStatementKey());
}
}
/**
* Creates a non-prepared statement.
*
* One-shot statements (i.e. non-prepared statements) must attach the db as soon as they
* are instantiated. The db is detached after executeUpdate or after executeQuery when
* its result set is closed.
*
* @param resultSetType a result set type; one of
* ResultSet.TYPE_FORWARD_ONLY
,
* ResultSet.TYPE_SCROLL_INSENSITIVE
, or
* ResultSet.TYPE_SCROLL_SENSITIVE
* @param resultSetConcurrency a concurrency type; one of
* ResultSet.CONCUR_READ_ONLY
or
* ResultSet.CONCUR_UPDATABLE
* @return a new Statement
object that will generate
* ResultSet
objects with the given type and
* concurrency
*/
public StatementWrapper createStatement (int resultSetType, int resultSetConcurrency) {
try {
Statement stmt = getConnection().createStatement(resultSetType, resultSetConcurrency);
return new StatementWrapper(this, stmt);
}
catch (SQLException ex) {
throw createFromSqlException("creating statement failed for " + this, ex);
}
}
/**
* Creates a prepared statement.
*
* @param statementKey the statement key, null if one-shot
* @param sql the sql code
* @param resultSetType the result set type
* @param resultSetConcurrency the result set concurrency
* @return the statement
*/
public PreparedStatementWrapper createPreparedStatement(StatementKey statementKey, String sql,
int resultSetType, int resultSetConcurrency) {
try {
sql = backend.optimizeSql(sql);
return new PreparedStatementWrapper(
this,
getConnection().prepareStatement(sql, resultSetType, resultSetConcurrency),
statementKey,
sql);
}
catch (SQLException e) {
throw createFromSqlException("creating prepared statement failed", e);
}
}
/**
* Gets a prepared statement.
*
* The statement will be reused if already prepared.
* Else it will be prepared according to the statement definition in BackendStatement.
*
* @param statementKey the unique statement key
* @param alwaysPrepare true if always do a physical prepare
* @param resultSetType one of ResultSet.TYPE_...
* @param resultSetConcurrency one of ResultSet.CONCUR_...
* @param sqlSupplier the SQL supplier
* @return the prepared statement for this connection
*/
public PreparedStatementWrapper getPreparedStatement (StatementKey statementKey, boolean alwaysPrepare,
int resultSetType, int resultSetConcurrency,
SqlSupplier sqlSupplier) {
if (statementKey == null) {
throw new PersistenceException("statement key required");
}
assertAttached();
PreparedStatementWrapper prepStmt = preparedStatements.get(statementKey);
if (prepStmt == null || prepStmt.isClosed() || alwaysPrepare) {
// we need to prepare it
prepStmt = createPreparedStatement(statementKey, sqlSupplier.get(backend).toString(), resultSetType, resultSetConcurrency);
preparedStatements.put(statementKey, prepStmt);
LOGGER.finer("statement {0} prepared on {1}", prepStmt, this);
}
else {
// already created: check that parameters are the same
if (checkStatementResultSetParameters) {
checkStatement(prepStmt, resultSetType, resultSetConcurrency);
}
LOGGER.finer("statement {0} re-used on {1}", prepStmt, this);
}
return prepStmt;
}
/**
* Checks for a dead communications link.
* If the link is down, for example the database server closed it,
* the connection is marked dead.
*
* @param ex the sql exception
* @return true if connection is dead
*/
public boolean checkForDeadLink(SQLException ex) {
if (!isDead() && backend.isCommunicationLinkException(ex)) {
// some severe comlink error, probably closed by server
setDead(true);
}
return isDead();
}
/**
* Creates a {@link PersistenceException} or {@link SessionClosedException} depending on the SQL error.
* If the session was reconnected, a {@link ReconnectedException} is returned.
*
* @param message the message
* @param sx the SQL exception
* @return the persistence exception
*/
public PersistenceException createFromSqlException(String message, SQLException sx) {
PersistenceException persistenceException;
if (checkForDeadLink(sx)) {
persistenceException = new SessionClosedException(db, message, sx);
Db ldb = db;
if (ldb != null && ldb.optionallyReconnect()) {
throw new ReconnectedException(persistenceException);
}
}
else {
persistenceException = new PersistenceException(db, message, sx);
}
return persistenceException;
}
/**
* Check that requested resulttype and concurrency matches the prepared statement to reuse.
*
* @param prepStmt the prepared statement
* @param resultSetType the requested result set type
* @param resultSetConcurrency the requested result set concurrency
*/
private void checkStatement(PreparedStatementWrapper prepStmt, int resultSetType, int resultSetConcurrency) {
Statement stmt = prepStmt.getStatement();
try {
if (stmt.getResultSetType() != resultSetType) {
throw new PersistenceException("wrong requested resultset type: " + resultSetType + " != statement: " +
stmt.getResultSetType() + " for '" + prepStmt.getSql() + "'");
}
if (stmt.getResultSetConcurrency() != resultSetConcurrency) {
throw new PersistenceException("wrong requested resultset concurrency: " + resultSetConcurrency + " != statement: " +
stmt.getResultSetConcurrency() + " for '" + prepStmt.getSql() + "'");
}
}
catch (SQLException sqx) {
throw createFromSqlException("cannot determine prepared statement configuration for '" + prepStmt.getSql() + "'", sqx);
}
}
private void restoreTransactionDefaults() {
if (txIsolation != defaultTxIsolation) {
setTransactionIsolation(defaultTxIsolation);
}
if (readOnly != defaultReadOnly) {
setReadOnly(defaultReadOnly);
}
}
}