All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.objectstyle.cayenne.conn.ConnectionWrapper Maven / Gradle / Ivy

/* ====================================================================
 * 
 * The ObjectStyle Group Software License, version 1.1
 * ObjectStyle Group - http://objectstyle.org/
 * 
 * Copyright (c) 2002-2005, Andrei (Andrus) Adamchik and individual authors
 * of the software. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 
 * 3. The end-user documentation included with the redistribution, if any,
 *    must include the following acknowlegement:
 *    "This product includes software developed by independent contributors
 *    and hosted on ObjectStyle Group web site (http://objectstyle.org/)."
 *    Alternately, this acknowlegement may appear in the software itself,
 *    if and wherever such third-party acknowlegements normally appear.
 * 
 * 4. The names "ObjectStyle Group" and "Cayenne" must not be used to endorse
 *    or promote products derived from this software without prior written
 *    permission. For written permission, email
 *    "andrus at objectstyle dot org".
 * 
 * 5. Products derived from this software may not be called "ObjectStyle"
 *    or "Cayenne", nor may "ObjectStyle" or "Cayenne" appear in their
 *    names without prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE OBJECTSTYLE GROUP OR
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * ====================================================================
 * 
 * This software consists of voluntary contributions made by many
 * individuals and hosted on ObjectStyle Group web site.  For more
 * information on the ObjectStyle Group, please see
 * .
 */
package org.objectstyle.cayenne.conn;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.SQLWarning;
import java.sql.Savepoint;
import java.sql.Statement;
import java.util.Map;

/**
 * ConnectionWrapper is a java.sql.Connection implementation that wraps
 * another Connection, delegating method calls to this connection. It works in conjunction
 * with PooledConnectionImpl, to generate pool events, provide limited automated
 * reconnection functionality, etc.
 * 
 * @author Andrei Adamchik
 */
public class ConnectionWrapper implements Connection {

    private Connection connection;
    private PooledConnectionImpl pooledConnection;
    private long lastReconnected;
    private int reconnectCount;

    /**
     * Fixes Sybase problems with autocommit. Used idea from Jonas
     * org.objectweb.jonas.jdbc_xa.ConnectionImpl (http://www.objectweb.org/jonas/).
     * 

* If problem is not the one that can be fixed by this patch, original exception is * rethrown. If exception occurs when fixing the problem, new exception is thrown. *

*/ static void sybaseAutoCommitPatch(Connection c, SQLException e, boolean autoCommit) throws SQLException { String s = e.getMessage().toLowerCase(); if (s.indexOf("set chained command not allowed") >= 0) { c.commit(); c.setAutoCommit(autoCommit); // Shouldn't fail now. } else { throw e; } } /** * Creates new ConnectionWrapper */ public ConnectionWrapper(Connection connection, PooledConnectionImpl pooledConnection) { this.connection = connection; this.pooledConnection = pooledConnection; } protected void reconnect(SQLException exception) throws SQLException { // if there was a relatively recent reconnect, just rethrow an error // and retire itself. THIS WILL PREVENT RECONNECT LOOPS if (reconnectCount > 0 && System.currentTimeMillis() - lastReconnected < 60000) { retire(exception); throw exception; } pooledConnection.reconnect(); // Pooled connection will wrap returned connection into // another ConnectionWrapper.... lets get the real connection // underneath... Connection connection = pooledConnection.getConnection(); if (connection instanceof ConnectionWrapper) { this.connection = ((ConnectionWrapper) connection).connection; } else { this.connection = connection; } lastReconnected = System.currentTimeMillis(); reconnectCount++; } protected void retire(SQLException exception) { // notify all the listeners.... pooledConnection.connectionErrorNotification(exception); } public void clearWarnings() throws SQLException { try { connection.clearWarnings(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void close() throws SQLException { if (null != pooledConnection) { pooledConnection.returnConnectionToThePool(); } connection = null; pooledConnection = null; } public void commit() throws SQLException { try { connection.commit(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public Statement createStatement() throws SQLException { try { return connection.createStatement(); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return createStatement(); } } public Statement createStatement(int resultSetType, int resultSetConcurrency) throws SQLException { try { return connection.createStatement(resultSetType, resultSetConcurrency); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return createStatement(resultSetType, resultSetConcurrency); } } public boolean getAutoCommit() throws SQLException { try { return connection.getAutoCommit(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public String getCatalog() throws SQLException { try { return connection.getCatalog(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public DatabaseMetaData getMetaData() throws SQLException { try { return connection.getMetaData(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public int getTransactionIsolation() throws SQLException { try { return connection.getTransactionIsolation(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public SQLWarning getWarnings() throws SQLException { try { return connection.getWarnings(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public boolean isClosed() throws SQLException { if (connection != null) { try { return connection.isClosed(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } else return true; } public boolean isReadOnly() throws SQLException { try { return connection.isReadOnly(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public String nativeSQL(String sql) throws SQLException { try { return connection.nativeSQL(sql); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public CallableStatement prepareCall(String sql) throws SQLException { try { return connection.prepareCall(sql); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareCall(sql); } } public CallableStatement prepareCall( String sql, int resultSetType, int resultSetConcurrency) throws SQLException { try { return connection.prepareCall(sql, resultSetType, resultSetConcurrency); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareCall(sql, resultSetType, resultSetConcurrency); } } public PreparedStatement prepareStatement(String sql) throws SQLException { try { return connection.prepareStatement(sql); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareStatement(sql); } } public PreparedStatement prepareStatement( String sql, int resultSetType, int resultSetConcurrency) throws SQLException { try { return connection.prepareStatement(sql, resultSetType, resultSetConcurrency); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareStatement(sql, resultSetType, resultSetConcurrency); } } public void rollback() throws SQLException { try { connection.rollback(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void setAutoCommit(boolean autoCommit) throws SQLException { try { connection.setAutoCommit(autoCommit); } catch (SQLException sqlEx) { try { // apply Sybase patch sybaseAutoCommitPatch(connection, sqlEx, autoCommit); } catch (SQLException patchEx) { retire(sqlEx); throw sqlEx; } } } public void setCatalog(String catalog) throws SQLException { try { connection.setCatalog(catalog); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void setReadOnly(boolean readOnly) throws SQLException { try { connection.setReadOnly(readOnly); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void setTransactionIsolation(int level) throws SQLException { try { connection.setTransactionIsolation(level); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public Map getTypeMap() throws SQLException { try { return connection.getTypeMap(); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void setTypeMap(Map map) throws SQLException { try { connection.setTypeMap(map); } catch (SQLException sqlEx) { retire(sqlEx); throw sqlEx; } } public void setHoldability(int holdability) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method setHoldability() not yet implemented."); } public int getHoldability() throws SQLException { throw new java.lang.UnsupportedOperationException( "Method getHoldability() not yet implemented."); } public Savepoint setSavepoint() throws SQLException { throw new java.lang.UnsupportedOperationException( "Method setSavepoint() not yet implemented."); } public Savepoint setSavepoint(String name) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method setSavepoint() not yet implemented."); } public void rollback(Savepoint savepoint) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method rollback() not yet implemented."); } public void releaseSavepoint(Savepoint savepoint) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method releaseSavepoint() not yet implemented."); } public Statement createStatement( int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method createStatement() not yet implemented."); } public PreparedStatement prepareStatement( String sql, int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { throw new java.lang.UnsupportedOperationException( "Method prepareStatement() not yet implemented."); } public CallableStatement prepareCall( String sql, int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { try { return connection.prepareCall( sql, resultSetType, resultSetConcurrency, resultSetHoldability); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareCall( sql, resultSetType, resultSetConcurrency, resultSetHoldability); } } public PreparedStatement prepareStatement(String sql, int autoGeneratedKeys) throws SQLException { try { return connection.prepareStatement(sql, autoGeneratedKeys); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareStatement(sql, autoGeneratedKeys); } } public PreparedStatement prepareStatement(String sql, int[] columnIndexes) throws SQLException { try { return connection.prepareStatement(sql, columnIndexes); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareStatement(sql, columnIndexes); } } public PreparedStatement prepareStatement(String sql, String[] columnNames) throws SQLException { try { return connection.prepareStatement(sql, columnNames); } catch (SQLException sqlEx) { // reconnect has code to prevent loops reconnect(sqlEx); return prepareStatement(sql, columnNames); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy