ManagedConnection.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.managed;

  18. import java.sql.Connection;
  19. import java.sql.SQLException;
  20. import java.util.concurrent.locks.Lock;
  21. import java.util.concurrent.locks.ReentrantLock;

  22. import org.apache.commons.dbcp2.DelegatingConnection;
  23. import org.apache.commons.pool2.ObjectPool;

  24. /**
  25.  * ManagedConnection is responsible for managing a database connection in a transactional environment (typically called
  26.  * "Container Managed"). A managed connection operates like any other connection when no global transaction (a.k.a. XA
  27.  * transaction or JTA Transaction) is in progress. When a global transaction is active a single physical connection to
  28.  * the database is used by all ManagedConnections accessed in the scope of the transaction. Connection sharing means
  29.  * that all data access during a transaction has a consistent view of the database. When the global transaction is
  30.  * committed or rolled back the enlisted connections are committed or rolled back. Typically, upon transaction
  31.  * completion, a connection returns to the auto commit setting in effect before being enlisted in the transaction, but
  32.  * some vendors do not properly implement this.
  33.  * <p>
  34.  * When enlisted in a transaction the setAutoCommit(), commit(), rollback(), and setReadOnly() methods throw a
  35.  * SQLException. This is necessary to assure that the transaction completes as a single unit.
  36.  * </p>
  37.  *
  38.  * @param <C>
  39.  *            the Connection type
  40.  *
  41.  * @since 2.0
  42.  */
  43. public class ManagedConnection<C extends Connection> extends DelegatingConnection<C> {

  44.     /**
  45.      * Delegates to {@link ManagedConnection#transactionComplete()} for transaction completion events.
  46.      *
  47.      * @since 2.0
  48.      */
  49.     protected class CompletionListener implements TransactionContextListener {
  50.         @Override
  51.         public void afterCompletion(final TransactionContext completedContext, final boolean committed) {
  52.             if (completedContext == transactionContext) {
  53.                 transactionComplete();
  54.             }
  55.         }
  56.     }

  57.     private final ObjectPool<C> pool;
  58.     private final TransactionRegistry transactionRegistry;
  59.     private final boolean accessToUnderlyingConnectionAllowed;
  60.     private TransactionContext transactionContext;
  61.     private boolean isSharedConnection;
  62.     private final Lock lock;

  63.     /**
  64.      * Constructs a new instance responsible for managing a database connection in a transactional environment.
  65.      *
  66.      * @param pool
  67.      *            The connection pool.
  68.      * @param transactionRegistry
  69.      *            The transaction registry.
  70.      * @param accessToUnderlyingConnectionAllowed
  71.      *            Whether or not to allow access to the underlying Connection.
  72.      * @throws SQLException
  73.      *             Thrown when there is problem managing transactions.
  74.      */
  75.     public ManagedConnection(final ObjectPool<C> pool, final TransactionRegistry transactionRegistry,
  76.             final boolean accessToUnderlyingConnectionAllowed) throws SQLException {
  77.         super(null);
  78.         this.pool = pool;
  79.         this.transactionRegistry = transactionRegistry;
  80.         this.accessToUnderlyingConnectionAllowed = accessToUnderlyingConnectionAllowed;
  81.         this.lock = new ReentrantLock();
  82.         updateTransactionStatus();
  83.     }

  84.     @Override
  85.     protected void checkOpen() throws SQLException {
  86.         super.checkOpen();
  87.         updateTransactionStatus();
  88.     }

  89.     @Override
  90.     public void close() throws SQLException {
  91.         if (!isClosedInternal()) {
  92.             // Don't actually close the connection if in a transaction. The
  93.             // connection will be closed by the transactionComplete method.
  94.             //
  95.             // DBCP-484 we need to make sure setClosedInternal(true) being
  96.             // invoked if transactionContext is not null as this value will
  97.             // be modified by the transactionComplete method which could run
  98.             // in the different thread with the transaction calling back.
  99.             lock.lock();
  100.             try {
  101.                 if (transactionContext == null || transactionContext.isTransactionComplete()) {
  102.                     super.close();
  103.                 }
  104.             } finally {
  105.                 try {
  106.                     setClosedInternal(true);
  107.                 } finally {
  108.                     lock.unlock();
  109.                 }
  110.             }
  111.         }
  112.     }

  113.     @Override
  114.     public void commit() throws SQLException {
  115.         if (transactionContext != null) {
  116.             throw new SQLException("Commit cannot be set while enrolled in a transaction");
  117.         }
  118.         super.commit();
  119.     }

  120.     @Override
  121.     public C getDelegate() {
  122.         if (isAccessToUnderlyingConnectionAllowed()) {
  123.             return getDelegateInternal();
  124.         }
  125.         return null;
  126.     }

  127.     //
  128.     // The following methods can't be used while enlisted in a transaction
  129.     //

  130.     @Override
  131.     public Connection getInnermostDelegate() {
  132.         if (isAccessToUnderlyingConnectionAllowed()) {
  133.             return super.getInnermostDelegateInternal();
  134.         }
  135.         return null;
  136.     }

  137.     /**
  138.      * Gets the transaction context.
  139.      *
  140.      * @return The transaction context.
  141.      * @since 2.6.0
  142.      */
  143.     public TransactionContext getTransactionContext() {
  144.         return transactionContext;
  145.     }

  146.     /**
  147.      * Gets the transaction registry.
  148.      *
  149.      * @return The transaction registry.
  150.      * @since 2.6.0
  151.      */
  152.     public TransactionRegistry getTransactionRegistry() {
  153.         return transactionRegistry;
  154.     }

  155.     /**
  156.      * If false, getDelegate() and getInnermostDelegate() will return null.
  157.      *
  158.      * @return if false, getDelegate() and getInnermostDelegate() will return null
  159.      */
  160.     public boolean isAccessToUnderlyingConnectionAllowed() {
  161.         return accessToUnderlyingConnectionAllowed;
  162.     }

  163.     @Override
  164.     public void rollback() throws SQLException {
  165.         if (transactionContext != null) {
  166.             throw new SQLException("Commit cannot be set while enrolled in a transaction");
  167.         }
  168.         super.rollback();
  169.     }

  170.     @Override
  171.     public void setAutoCommit(final boolean autoCommit) throws SQLException {
  172.         if (transactionContext != null) {
  173.             throw new SQLException("Auto-commit cannot be set while enrolled in a transaction");
  174.         }
  175.         super.setAutoCommit(autoCommit);
  176.     }

  177.     @Override
  178.     public void setReadOnly(final boolean readOnly) throws SQLException {
  179.         if (transactionContext != null) {
  180.             throw new SQLException("Read-only cannot be set while enrolled in a transaction");
  181.         }
  182.         super.setReadOnly(readOnly);
  183.     }

  184.     /**
  185.      * Completes the transaction.
  186.      */
  187.     protected void transactionComplete() {
  188.         lock.lock();
  189.         try {
  190.             transactionContext.completeTransaction();
  191.         } finally {
  192.             lock.unlock();
  193.         }

  194.         // If we were using a shared connection, clear the reference now that
  195.         // the transaction has completed
  196.         if (isSharedConnection) {
  197.             setDelegate(null);
  198.             isSharedConnection = false;
  199.         }

  200.         // autoCommit may have been changed directly on the underlying connection
  201.         clearCachedState();

  202.         // If this connection was closed during the transaction and there is
  203.         // still a delegate present close it
  204.         final Connection delegate = getDelegateInternal();
  205.         if (isClosedInternal() && delegate != null) {
  206.             try {
  207.                 setDelegate(null);

  208.                 if (!delegate.isClosed()) {
  209.                     delegate.close();
  210.                 }
  211.             } catch (final SQLException ignored) {
  212.                 // Not a whole lot we can do here as connection is closed
  213.                 // and this is a transaction callback so there is no
  214.                 // way to report the error.
  215.             }
  216.         }
  217.     }

  218.     private void updateTransactionStatus() throws SQLException {
  219.         // if there is an active transaction context, assure the transaction context hasn't changed
  220.         if (transactionContext != null && !transactionContext.isTransactionComplete()) {
  221.             if (transactionContext.isActive()) {
  222.                 if (transactionContext != transactionRegistry.getActiveTransactionContext()) {
  223.                     throw new SQLException("Connection cannot be used while enlisted in another transaction");
  224.                 }
  225.                 return;
  226.             }
  227.             // transaction should have been cleared up by TransactionContextListener, but in
  228.             // rare cases another lister could have registered which uses the connection before
  229.             // our listener is called. In that rare case, trigger the transaction complete call now
  230.             transactionComplete();
  231.         }

  232.         // the existing transaction context ended (or we didn't have one), get the active transaction context
  233.         transactionContext = transactionRegistry.getActiveTransactionContext();

  234.         // if there is an active transaction context, and it already has a shared connection, use it
  235.         if (transactionContext != null && transactionContext.getSharedConnection() != null) {
  236.             // A connection for the connection factory has already been enrolled
  237.             // in the transaction, replace our delegate with the enrolled connection

  238.             // return current connection to the pool
  239.             @SuppressWarnings("resource")
  240.             final C connection = getDelegateInternal();
  241.             setDelegate(null);
  242.             if (connection != null && transactionContext.getSharedConnection() != connection) {
  243.                 try {
  244.                     pool.returnObject(connection);
  245.                 } catch (final Exception e) {
  246.                     // whatever... try to invalidate the connection
  247.                     try {
  248.                         pool.invalidateObject(connection);
  249.                     } catch (final Exception ignored) {
  250.                         // no big deal
  251.                     }
  252.                 }
  253.             }

  254.             // add a listener to the transaction context
  255.             transactionContext.addTransactionContextListener(new CompletionListener());

  256.             // Set our delegate to the shared connection. Note that this will
  257.             // always be of type C since it has been shared by another
  258.             // connection from the same pool.
  259.             @SuppressWarnings("unchecked")
  260.             final C shared = (C) transactionContext.getSharedConnection();
  261.             setDelegate(shared);

  262.             // remember that we are using a shared connection, so it can be cleared after the
  263.             // transaction completes
  264.             isSharedConnection = true;
  265.         } else {
  266.             C connection = getDelegateInternal();
  267.             // if our delegate is null, create one
  268.             if (connection == null) {
  269.                 try {
  270.                     // borrow a new connection from the pool
  271.                     connection = pool.borrowObject();
  272.                     setDelegate(connection);
  273.                 } catch (final Exception e) {
  274.                     throw new SQLException("Unable to acquire a new connection from the pool", e);
  275.                 }
  276.             }

  277.             // if we have a transaction, out delegate becomes the shared delegate
  278.             if (transactionContext != null) {
  279.                 // add a listener to the transaction context
  280.                 transactionContext.addTransactionContextListener(new CompletionListener());

  281.                 // register our connection as the shared connection
  282.                 try {
  283.                     transactionContext.setSharedConnection(connection);
  284.                 } catch (final SQLException e) {
  285.                     // transaction is hosed
  286.                     transactionContext = null;
  287.                     try {
  288.                         pool.invalidateObject(connection);
  289.                     } catch (final Exception ignored) {
  290.                         // we are try but no luck
  291.                     }
  292.                     throw e;
  293.                 }
  294.             }
  295.         }
  296.         // autoCommit may have been changed directly on the underlying
  297.         // connection
  298.         clearCachedState();
  299.     }
  300. }