001    /*
002     * Licensed to the Apache Software Foundation (ASF) under one or more
003     * contributor license agreements.  See the NOTICE file distributed with
004     * this work for additional information regarding copyright ownership.
005     * The ASF licenses this file to You under the Apache License, Version 2.0
006     * (the "License"); you may not use this file except in compliance with
007     * the License.  You may obtain a copy of the License at
008     * 
009     *      http://www.apache.org/licenses/LICENSE-2.0
010     * 
011     * Unless required by applicable law or agreed to in writing, software
012     * distributed under the License is distributed on an "AS IS" BASIS,
013     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014     * See the License for the specific language governing permissions and
015     * limitations under the License.
016     */
017    
018    package org.apache.commons.dbcp;
019    
020    import java.sql.Connection;
021    import java.sql.PreparedStatement;
022    import java.sql.ResultSet;
023    import java.sql.SQLException;
024    import java.util.List;
025    
026    import org.apache.commons.pool.KeyedObjectPool;
027    
028    /**
029     * A {@link DelegatingPreparedStatement} that cooperates with
030     * {@link PoolingConnection} to implement a pool of {@link PreparedStatement}s.
031     * <p>
032     * My {@link #close} method returns me to my containing pool. (See {@link PoolingConnection}.)
033     *
034     * @see PoolingConnection
035     * @author Rodney Waldhoff
036     * @author Glenn L. Nielsen
037     * @author James House
038     * @author Dirk Verbeeck
039     * @version $Revision: 745860 $ $Date: 2009-02-19 08:45:07 -0500 (Thu, 19 Feb 2009) $
040     */
041    public class PoolablePreparedStatement extends DelegatingPreparedStatement implements PreparedStatement {
042        /**
043         * The {@link KeyedObjectPool} from which I was obtained.
044         */
045        protected KeyedObjectPool _pool = null;
046    
047        /**
048         * My "key" as used by {@link KeyedObjectPool}.
049         */
050        protected Object _key = null;
051    
052        private volatile boolean batchAdded = false;
053    
054        /**
055         * Constructor
056         * @param stmt my underlying {@link PreparedStatement}
057         * @param key my key" as used by {@link KeyedObjectPool}
058         * @param pool the {@link KeyedObjectPool} from which I was obtained.
059         * @param conn the {@link Connection} from which I was created
060         */
061        public PoolablePreparedStatement(PreparedStatement stmt, Object key, KeyedObjectPool pool, Connection conn) {
062            super((DelegatingConnection) conn, stmt);
063            _pool = pool;
064            _key = key;
065    
066            // Remove from trace now because this statement will be 
067            // added by the activate method.
068            if(_conn != null) {
069                _conn.removeTrace(this);
070            }
071        }
072    
073        /**
074         * Add batch.
075         */
076        public void addBatch() throws SQLException {
077            super.addBatch();
078            batchAdded = true;
079        }
080    
081        /**
082         * Clear Batch.
083         */
084        public void clearBatch() throws SQLException {
085            batchAdded = false;
086            super.clearBatch();
087        }
088    
089        /**
090         * Return me to my pool.
091         */
092        public void close() throws SQLException {
093            // calling close twice should have no effect
094            if (!isClosed()) {
095                try {
096                    _pool.returnObject(_key,this);
097                } catch(SQLException e) {
098                    throw e;
099                } catch(RuntimeException e) {
100                    throw e;
101                } catch(Exception e) {
102                    throw new SQLNestedException("Cannot close preparedstatement (return to pool failed)", e);
103                }
104            }
105        }
106        
107        protected void activate() throws SQLException{
108            _closed = false;
109            if(_conn != null) {
110                _conn.addTrace(this);
111            }
112            super.activate();
113        }
114      
115        protected void passivate() throws SQLException {
116            _closed = true;
117            if(_conn != null) {
118                _conn.removeTrace(this);
119            }
120    
121            // The JDBC spec requires that a statment close any open
122            // ResultSet's when it is closed.
123            // FIXME The PreparedStatement we're wrapping should handle this for us.
124            // See bug 17301 for what could happen when ResultSets are closed twice.
125            List resultSets = getTrace();
126            if( resultSets != null) {
127                ResultSet[] set = (ResultSet[]) resultSets.toArray(new ResultSet[resultSets.size()]);
128                for (int i = 0; i < set.length; i++) {
129                    set[i].close();
130                }
131                clearTrace();
132            }
133            if (batchAdded) {
134                clearBatch();
135            }
136            
137            super.passivate();
138        }
139    
140    }