ConnectionImpl.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.dbcp2.cpdsadapter;

  18. import java.sql.CallableStatement;
  19. import java.sql.Connection;
  20. import java.sql.PreparedStatement;
  21. import java.sql.SQLException;

  22. import org.apache.commons.dbcp2.DelegatingCallableStatement;
  23. import org.apache.commons.dbcp2.DelegatingConnection;
  24. import org.apache.commons.dbcp2.DelegatingPreparedStatement;

  25. /**
  26.  * This class is the {@code Connection} that will be returned from
  27.  * {@code PooledConnectionImpl.getConnection()}. Most methods are wrappers around the JDBC 1.x
  28.  * {@code Connection}. A few exceptions include preparedStatement and close. In accordance with the JDBC
  29.  * specification this Connection cannot be used after closed() is called. Any further usage will result in an
  30.  * SQLException.
  31.  * <p>
  32.  * ConnectionImpl extends DelegatingConnection to enable access to the underlying connection.
  33.  * </p>
  34.  *
  35.  * @since 2.0
  36.  */
  37. final class ConnectionImpl extends DelegatingConnection<Connection> {

  38.     private final boolean accessToUnderlyingConnectionAllowed;

  39.     /** The object that instantiated this object */
  40.     private final PooledConnectionImpl pooledConnection;

  41.     /**
  42.      * Creates a {@code ConnectionImpl}.
  43.      *
  44.      * @param pooledConnection
  45.      *            The PooledConnection that is calling the ctor.
  46.      * @param connection
  47.      *            The JDBC 1.x Connection to wrap.
  48.      * @param accessToUnderlyingConnectionAllowed
  49.      *            if true, then access is allowed to the underlying connection
  50.      */
  51.     ConnectionImpl(final PooledConnectionImpl pooledConnection, final Connection connection,
  52.             final boolean accessToUnderlyingConnectionAllowed) {
  53.         super(connection);
  54.         this.pooledConnection = pooledConnection;
  55.         this.accessToUnderlyingConnectionAllowed = accessToUnderlyingConnectionAllowed;
  56.     }

  57.     /**
  58.      * Marks the Connection as closed, and notifies the pool that the pooled connection is available.
  59.      * <p>
  60.      * In accordance with the JDBC specification this Connection cannot be used after closed() is called. Any further
  61.      * usage will result in an SQLException.
  62.      * </p>
  63.      *
  64.      * @throws SQLException
  65.      *             The database connection couldn't be closed.
  66.      */
  67.     @Override
  68.     public void close() throws SQLException {
  69.         if (!isClosedInternal()) {
  70.             try {
  71.                 passivate();
  72.             } finally {
  73.                 setClosedInternal(true);
  74.                 pooledConnection.notifyListeners();
  75.             }
  76.         }
  77.     }

  78.     /**
  79.      * Gets the delegated connection, if allowed.
  80.      *
  81.      * @return the internal connection, or null if access is not allowed.
  82.      * @see #isAccessToUnderlyingConnectionAllowed()
  83.      */
  84.     @Override
  85.     public Connection getDelegate() {
  86.         if (isAccessToUnderlyingConnectionAllowed()) {
  87.             return getDelegateInternal();
  88.         }
  89.         return null;
  90.     }

  91.     /**
  92.      * Gets the innermost connection, if allowed.
  93.      *
  94.      * @return the innermost internal connection, or null if access is not allowed.
  95.      * @see #isAccessToUnderlyingConnectionAllowed()
  96.      */
  97.     @Override
  98.     public Connection getInnermostDelegate() {
  99.         if (isAccessToUnderlyingConnectionAllowed()) {
  100.             return super.getInnermostDelegateInternal();
  101.         }
  102.         return null;
  103.     }

  104.     /**
  105.      * Package-private for tests.
  106.      *
  107.      * @return the PooledConnectionImpl.
  108.      */
  109.     PooledConnectionImpl getPooledConnectionImpl() {
  110.         return pooledConnection;
  111.     }

  112.     /**
  113.      * If false, getDelegate() and getInnermostDelegate() will return null.
  114.      *
  115.      * @return true if access is allowed to the underlying connection
  116.      * @see ConnectionImpl
  117.      */
  118.     public boolean isAccessToUnderlyingConnectionAllowed() {
  119.         return accessToUnderlyingConnectionAllowed;
  120.     }

  121.     /**
  122.      * If pooling of {@code CallableStatement}s is turned on in the {@link DriverAdapterCPDS}, a pooled object may
  123.      * be returned, otherwise delegate to the wrapped JDBC 1.x {@link java.sql.Connection}.
  124.      *
  125.      * @param sql
  126.      *            an SQL statement that may contain one or more '?' parameter placeholders. Typically, this statement is
  127.      *            specified using JDBC call escape syntax.
  128.      * @return a default {@code CallableStatement} object containing the pre-compiled SQL statement.
  129.      * @throws SQLException
  130.      *                Thrown if a database access error occurs or this method is called on a closed connection.
  131.      * @since 2.4.0
  132.      */
  133.     @Override
  134.     public CallableStatement prepareCall(final String sql) throws SQLException {
  135.         checkOpen();
  136.         try {
  137.             return new DelegatingCallableStatement(this, pooledConnection.prepareCall(sql));
  138.         } catch (final SQLException e) {
  139.             handleException(e); // Does not return
  140.             return null;
  141.         }
  142.     }

  143.     /**
  144.      * If pooling of {@code CallableStatement}s is turned on in the {@link DriverAdapterCPDS}, a pooled object may
  145.      * be returned, otherwise delegate to the wrapped JDBC 1.x {@link java.sql.Connection}.
  146.      *
  147.      * @param sql
  148.      *            a {@code String} object that is the SQL statement to be sent to the database; may contain on or
  149.      *            more '?' parameters.
  150.      * @param resultSetType
  151.      *            a result set type; one of {@code ResultSet.TYPE_FORWARD_ONLY},
  152.      *            {@code ResultSet.TYPE_SCROLL_INSENSITIVE}, or {@code ResultSet.TYPE_SCROLL_SENSITIVE}.
  153.      * @param resultSetConcurrency
  154.      *            a concurrency type; one of {@code ResultSet.CONCUR_READ_ONLY} or
  155.      *            {@code ResultSet.CONCUR_UPDATABLE}.
  156.      * @return a {@code CallableStatement} object containing the pre-compiled SQL statement that will produce
  157.      *         {@code ResultSet} objects with the given type and concurrency.
  158.      * @throws SQLException
  159.      *             Thrown if a database access error occurs, this method is called on a closed connection or the given
  160.      *             parameters are not {@code ResultSet} constants indicating type and concurrency.
  161.      * @since 2.4.0
  162.      */
  163.     @Override
  164.     public CallableStatement prepareCall(final String sql, final int resultSetType, final int resultSetConcurrency)
  165.             throws SQLException {
  166.         checkOpen();
  167.         try {
  168.             return new DelegatingCallableStatement(this,
  169.                     pooledConnection.prepareCall(sql, resultSetType, resultSetConcurrency));
  170.         } catch (final SQLException e) {
  171.             handleException(e); // Does not return
  172.             return null;
  173.         }
  174.     }

  175.     /**
  176.      * If pooling of {@code CallableStatement}s is turned on in the {@link DriverAdapterCPDS}, a pooled object may
  177.      * be returned, otherwise delegate to the wrapped JDBC 1.x {@link java.sql.Connection}.
  178.      *
  179.      * @param sql
  180.      *            a {@code String} object that is the SQL statement to be sent to the database; may contain on or
  181.      *            more '?' parameters.
  182.      * @param resultSetType
  183.      *            one of the following {@code ResultSet} constants: {@code ResultSet.TYPE_FORWARD_ONLY},
  184.      *            {@code ResultSet.TYPE_SCROLL_INSENSITIVE}, or {@code ResultSet.TYPE_SCROLL_SENSITIVE}.
  185.      * @param resultSetConcurrency
  186.      *            one of the following {@code ResultSet} constants: {@code ResultSet.CONCUR_READ_ONLY} or
  187.      *            {@code ResultSet.CONCUR_UPDATABLE}.
  188.      * @param resultSetHoldability
  189.      *            one of the following {@code ResultSet} constants: {@code ResultSet.HOLD_CURSORS_OVER_COMMIT}
  190.      *            or {@code ResultSet.CLOSE_CURSORS_AT_COMMIT}.
  191.      * @return a new {@code CallableStatement} object, containing the pre-compiled SQL statement, that will
  192.      *         generate {@code ResultSet} objects with the given type, concurrency, and holdability.
  193.      * @throws SQLException
  194.      *             Thrown if a database access error occurs, this method is called on a closed connection or the given
  195.      *             parameters are not {@code ResultSet} constants indicating type, concurrency, and holdability.
  196.      * @since 2.4.0
  197.      */
  198.     @Override
  199.     public CallableStatement prepareCall(final String sql, final int resultSetType, final int resultSetConcurrency,
  200.             final int resultSetHoldability) throws SQLException {
  201.         checkOpen();
  202.         try {
  203.             return new DelegatingCallableStatement(this,
  204.                     pooledConnection.prepareCall(sql, resultSetType, resultSetConcurrency, resultSetHoldability));
  205.         } catch (final SQLException e) {
  206.             handleException(e); // Does not return
  207.             return null;
  208.         }
  209.     }

  210.     /**
  211.      * If pooling of {@code PreparedStatement}s is turned on in the {@link DriverAdapterCPDS}, a pooled object may
  212.      * be returned, otherwise delegate to the wrapped JDBC 1.x {@link java.sql.Connection}.
  213.      *
  214.      * @param sql
  215.      *            SQL statement to be prepared
  216.      * @return the prepared statement
  217.      * @throws SQLException
  218.      *             if this connection is closed or an error occurs in the wrapped connection.
  219.      */
  220.     @Override
  221.     public PreparedStatement prepareStatement(final String sql) throws SQLException {
  222.         checkOpen();
  223.         try {
  224.             return new DelegatingPreparedStatement(this, pooledConnection.prepareStatement(sql));
  225.         } catch (final SQLException e) {
  226.             handleException(e); // Does not return
  227.             return null;
  228.         }
  229.     }

  230.     @Override
  231.     public PreparedStatement prepareStatement(final String sql, final int autoGeneratedKeys) throws SQLException {
  232.         checkOpen();
  233.         try {
  234.             return new DelegatingPreparedStatement(this, pooledConnection.prepareStatement(sql, autoGeneratedKeys));
  235.         } catch (final SQLException e) {
  236.             handleException(e);
  237.             return null;
  238.         }
  239.     }

  240.     //
  241.     // Methods for accessing the delegate connection
  242.     //

  243.     /**
  244.      * If pooling of {@code PreparedStatement}s is turned on in the {@link DriverAdapterCPDS}, a pooled object may
  245.      * be returned, otherwise delegate to the wrapped JDBC 1.x {@link java.sql.Connection}.
  246.      *
  247.      * @throws SQLException
  248.      *             if this connection is closed or an error occurs in the wrapped connection.
  249.      */
  250.     @Override
  251.     public PreparedStatement prepareStatement(final String sql, final int resultSetType, final int resultSetConcurrency)
  252.             throws SQLException {
  253.         checkOpen();
  254.         try {
  255.             return new DelegatingPreparedStatement(this,
  256.                     pooledConnection.prepareStatement(sql, resultSetType, resultSetConcurrency));
  257.         } catch (final SQLException e) {
  258.             handleException(e);
  259.             return null;
  260.         }
  261.     }

  262.     @Override
  263.     public PreparedStatement prepareStatement(final String sql, final int resultSetType, final int resultSetConcurrency,
  264.             final int resultSetHoldability) throws SQLException {
  265.         checkOpen();
  266.         try {
  267.             return new DelegatingPreparedStatement(this,
  268.                     pooledConnection.prepareStatement(sql, resultSetType, resultSetConcurrency, resultSetHoldability));
  269.         } catch (final SQLException e) {
  270.             handleException(e);
  271.             return null;
  272.         }
  273.     }

  274.     @Override
  275.     public PreparedStatement prepareStatement(final String sql, final int[] columnIndexes) throws SQLException {
  276.         checkOpen();
  277.         try {
  278.             return new DelegatingPreparedStatement(this, pooledConnection.prepareStatement(sql, columnIndexes));
  279.         } catch (final SQLException e) {
  280.             handleException(e);
  281.             return null;
  282.         }
  283.     }

  284.     @Override
  285.     public PreparedStatement prepareStatement(final String sql, final String[] columnNames) throws SQLException {
  286.         checkOpen();
  287.         try {
  288.             return new DelegatingPreparedStatement(this, pooledConnection.prepareStatement(sql, columnNames));
  289.         } catch (final SQLException e) {
  290.             handleException(e);
  291.             return null;
  292.         }
  293.     }

  294. }