/vt-ldap/trunk/src/main/java/edu/vt/middleware/ldap/pool/BlockingConnectionPool.java

http://vt-middleware.googlecode.com/ · Java · 274 lines · 173 code · 24 blank · 77 comment · 23 complexity · cf03a06df44d3d3285231dca64b0bfdc MD5 · raw file

  1. /*
  2. $Id: BlockingConnectionPool.java 2197 2012-01-02 03:40:30Z dfisher $
  3. Copyright (C) 2003-2012 Virginia Tech.
  4. All rights reserved.
  5. SEE LICENSE FOR MORE INFORMATION
  6. Author: Middleware Services
  7. Email: middleware@vt.edu
  8. Version: $Revision: 2197 $
  9. Updated: $Date: 2012-01-02 04:40:30 +0100 (Mon, 02 Jan 2012) $
  10. */
  11. package edu.vt.middleware.ldap.pool;
  12. import java.util.NoSuchElementException;
  13. import java.util.concurrent.TimeUnit;
  14. import edu.vt.middleware.ldap.Connection;
  15. import edu.vt.middleware.ldap.DefaultConnectionFactory;
  16. /**
  17. * Implements a pool of connections that has a set minimum and maximum size. The
  18. * pool will not grow beyond the maximum size and when the pool is exhausted,
  19. * requests for new connections will block. The length of time the pool will
  20. * block is determined by {@link #getBlockWaitTime()}. By default the pool will
  21. * block indefinitely and there is no guarantee that waiting threads will be
  22. * serviced in the order in which they made their request. This implementation
  23. * should be used when you need to control the <em>exact</em> number of
  24. * connections that can be created. See {@link AbstractConnectionPool}.
  25. *
  26. * @author Middleware Services
  27. * @version $Revision: 2197 $ $Date: 2012-01-02 04:40:30 +0100 (Mon, 02 Jan 2012) $
  28. */
  29. public class BlockingConnectionPool extends AbstractConnectionPool
  30. implements ConnectionPool
  31. {
  32. /** Time in milliseconds to wait for an available connection. */
  33. private long blockWaitTime;
  34. /** Creates a new blocking pool. */
  35. public BlockingConnectionPool() {}
  36. /**
  37. * Creates a new blocking pool. The pool config is initialized with the
  38. * default values.
  39. *
  40. * @param cf connection factory
  41. */
  42. public BlockingConnectionPool(final DefaultConnectionFactory cf)
  43. {
  44. this(new PoolConfig(), cf);
  45. }
  46. /**
  47. * Creates a new blocking pool.
  48. *
  49. * @param pc pool configuration
  50. * @param cf connection factory
  51. */
  52. public BlockingConnectionPool(
  53. final PoolConfig pc,
  54. final DefaultConnectionFactory cf)
  55. {
  56. setPoolConfig(pc);
  57. setConnectionFactory(cf);
  58. }
  59. /**
  60. * Returns the block wait time. Default time is 0, which will wait
  61. * indefinitely.
  62. *
  63. * @return time in milliseconds to wait for available connections
  64. */
  65. public long getBlockWaitTime()
  66. {
  67. return blockWaitTime;
  68. }
  69. /**
  70. * Sets the block wait time. Default time is 0, which will wait indefinitely.
  71. *
  72. * @param time in milliseconds to wait for available connections
  73. */
  74. public void setBlockWaitTime(final long time)
  75. {
  76. if (time >= 0) {
  77. blockWaitTime = time;
  78. }
  79. }
  80. /** {@inheritDoc} */
  81. @Override
  82. public Connection getConnection()
  83. throws PoolException
  84. {
  85. PooledConnectionHandler pc = null;
  86. boolean create = false;
  87. logger.trace(
  88. "waiting on pool lock for check out {}",
  89. poolLock.getQueueLength());
  90. poolLock.lock();
  91. try {
  92. // if an available connection exists, use it
  93. // if no available connections and the pool can grow, attempt to create
  94. // otherwise the pool is full, block until a connection is returned
  95. if (available.size() > 0) {
  96. try {
  97. logger.trace("retrieve available connection");
  98. pc = retrieveAvailableConnection();
  99. } catch (NoSuchElementException e) {
  100. logger.error("could not remove connection from list", e);
  101. throw new IllegalStateException("Pool is empty", e);
  102. }
  103. } else if (active.size() < getPoolConfig().getMaxPoolSize()) {
  104. logger.trace("pool can grow, attempt to create connection");
  105. create = true;
  106. } else {
  107. logger.trace("pool is full, block until connection is available");
  108. pc = blockAvailableConnection();
  109. }
  110. } finally {
  111. poolLock.unlock();
  112. }
  113. if (create) {
  114. // previous block determined a creation should occur
  115. // block here until create occurs without locking the whole pool
  116. // if the pool is already maxed or creates are failing,
  117. // block until a connection is available
  118. checkOutLock.lock();
  119. try {
  120. boolean b = true;
  121. poolLock.lock();
  122. try {
  123. if (
  124. available.size() + active.size() ==
  125. getPoolConfig().getMaxPoolSize()) {
  126. b = false;
  127. }
  128. } finally {
  129. poolLock.unlock();
  130. }
  131. if (b) {
  132. pc = createActiveConnection();
  133. logger.trace("created new active connection: {}", pc);
  134. }
  135. } finally {
  136. checkOutLock.unlock();
  137. }
  138. if (pc == null) {
  139. logger.debug("create failed, block until connection is available");
  140. pc = blockAvailableConnection();
  141. }
  142. }
  143. if (pc != null) {
  144. activateAndValidateConnection(pc);
  145. } else {
  146. logger.error("Could not service check out request");
  147. throw new PoolExhaustedException(
  148. "Pool is empty and connection creation failed");
  149. }
  150. return createConnectionProxy(pc);
  151. }
  152. /**
  153. * Attempts to retrieve a connection from the available queue.
  154. *
  155. * @return connection from the pool
  156. *
  157. * @throws NoSuchElementException if the available queue is empty
  158. */
  159. protected PooledConnectionHandler retrieveAvailableConnection()
  160. {
  161. PooledConnectionHandler pc = null;
  162. logger.trace(
  163. "waiting on pool lock for retrieve available {}",
  164. poolLock.getQueueLength());
  165. poolLock.lock();
  166. try {
  167. pc = available.remove();
  168. active.add(pc);
  169. logger.trace("retrieved available connection: {}", pc);
  170. } finally {
  171. poolLock.unlock();
  172. }
  173. return pc;
  174. }
  175. /**
  176. * This blocks until a connection can be acquired.
  177. *
  178. * @return connection from the pool
  179. *
  180. * @throws PoolException if this method fails
  181. * @throws BlockingTimeoutException if this pool is configured with a block
  182. * time and it occurs
  183. * @throws PoolInterruptedException if the current thread is interrupted
  184. */
  185. protected PooledConnectionHandler blockAvailableConnection()
  186. throws PoolException
  187. {
  188. PooledConnectionHandler pc = null;
  189. logger.trace(
  190. "waiting on pool lock for block available {}",
  191. poolLock.getQueueLength());
  192. poolLock.lock();
  193. try {
  194. while (pc == null) {
  195. logger.trace("available pool is empty, waiting...");
  196. if (blockWaitTime > 0) {
  197. if (!poolNotEmpty.await(blockWaitTime, TimeUnit.MILLISECONDS)) {
  198. logger.debug("block time exceeded, throwing exception");
  199. throw new BlockingTimeoutException("Block time exceeded");
  200. }
  201. } else {
  202. poolNotEmpty.await();
  203. }
  204. logger.trace("notified to continue...");
  205. try {
  206. pc = retrieveAvailableConnection();
  207. } catch (NoSuchElementException e) {
  208. logger.trace("notified to continue but pool was empty");
  209. }
  210. }
  211. } catch (InterruptedException e) {
  212. logger.error("waiting for available connection interrupted", e);
  213. throw new PoolInterruptedException(
  214. "Interrupted while waiting for an available connection",
  215. e);
  216. } finally {
  217. poolLock.unlock();
  218. }
  219. return pc;
  220. }
  221. /** {@inheritDoc} */
  222. @Override
  223. public void putConnection(final Connection c)
  224. {
  225. final PooledConnectionHandler pc = retrieveInvocationHandler(c);
  226. final boolean valid = validateAndPassivateConnection(pc);
  227. logger.trace(
  228. "waiting on pool lock for check in {}",
  229. poolLock.getQueueLength());
  230. poolLock.lock();
  231. try {
  232. if (active.remove(pc)) {
  233. if (valid) {
  234. available.add(pc);
  235. logger.trace("returned active connection: {}", pc);
  236. poolNotEmpty.signal();
  237. }
  238. } else if (available.contains(pc)) {
  239. logger.warn("returned available connection: {}", pc);
  240. } else {
  241. logger.warn("attempt to return unknown connection: {}", pc);
  242. }
  243. } finally {
  244. poolLock.unlock();
  245. }
  246. }
  247. }