/vt-ldap/branches/vt-ldap-3/src/main/java/edu/vt/middleware/ldap/pool/BlockingLdapPool.java

http://vt-middleware.googlecode.com/ · Java · 323 lines · 222 code · 24 blank · 77 comment · 48 complexity · cf1b8a9ab1fc3769bf179037e28fe6e6 MD5 · raw file

  1. /*
  2. $Id: BlockingLdapPool.java 2241 2012-02-07 20:08:51Z dfisher $
  3. Copyright (C) 2003-2010 Virginia Tech.
  4. All rights reserved.
  5. SEE LICENSE FOR MORE INFORMATION
  6. Author: Middleware Services
  7. Email: middleware@vt.edu
  8. Version: $Revision: 2241 $
  9. Updated: $Date: 2012-02-07 21:08:51 +0100 (Tue, 07 Feb 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.Ldap;
  15. /**
  16. * <code>BlockingLdapPool</code> implements a pool of ldap objects that has a
  17. * set minimum and maximum size. The pool will not grow beyond the maximum size
  18. * and when the pool is exhausted, requests for new objects will block. The
  19. * length of time the pool will block is determined by {@link
  20. * #getBlockWaitTime()}. By default the pool will block indefinitely and there
  21. * is no guarantee that waiting threads will be serviced in the order in which
  22. * they made their request. This implementation should be used when you need to
  23. * control the <em>exact</em> number of ldap connections that can be created.
  24. * See {@link AbstractLdapPool}.
  25. *
  26. * @author Middleware Services
  27. * @version $Revision: 2241 $ $Date: 2012-02-07 21:08:51 +0100 (Tue, 07 Feb 2012) $
  28. */
  29. public class BlockingLdapPool extends AbstractLdapPool<Ldap>
  30. {
  31. /** Time in milliseconds to wait for an available ldap object. */
  32. private long blockWaitTime;
  33. /** Creates a new ldap pool using {@link DefaultLdapFactory}. */
  34. public BlockingLdapPool()
  35. {
  36. super(new LdapPoolConfig(), new DefaultLdapFactory());
  37. }
  38. /**
  39. * Creates a new ldap pool with the supplied ldap factory.
  40. *
  41. * @param lf ldap factory
  42. */
  43. public BlockingLdapPool(final LdapFactory<Ldap> lf)
  44. {
  45. super(new LdapPoolConfig(), lf);
  46. }
  47. /**
  48. * Creates a new ldap pool with the supplied ldap config and factory.
  49. *
  50. * @param lpc ldap pool configuration
  51. * @param lf ldap factory
  52. */
  53. public BlockingLdapPool(final LdapPoolConfig lpc, final LdapFactory<Ldap> lf)
  54. {
  55. super(lpc, lf);
  56. }
  57. /**
  58. * Returns the block wait time. Default time is 0, which will wait
  59. * indefinitely.
  60. *
  61. * @return time in milliseconds to wait for available ldap objects
  62. */
  63. public long getBlockWaitTime()
  64. {
  65. return this.blockWaitTime;
  66. }
  67. /**
  68. * Sets the block wait time. Default time is 0, which will wait indefinitely.
  69. *
  70. * @param time in milliseconds to wait for available ldap objects
  71. */
  72. public void setBlockWaitTime(final long time)
  73. {
  74. if (time >= 0) {
  75. this.blockWaitTime = time;
  76. }
  77. }
  78. /** {@inheritDoc} */
  79. public Ldap checkOut()
  80. throws LdapPoolException
  81. {
  82. Ldap l = null;
  83. boolean create = false;
  84. if (this.logger.isTraceEnabled()) {
  85. this.logger.trace(
  86. "waiting on pool lock for check out " + this.poolLock.getQueueLength());
  87. }
  88. this.poolLock.lock();
  89. try {
  90. // if an available object exists, use it
  91. // if no available objects and the pool can grow, attempt to create
  92. // otherwise the pool is full, block until an object is returned
  93. if (this.available.size() > 0) {
  94. try {
  95. if (this.logger.isTraceEnabled()) {
  96. this.logger.trace("retrieve available ldap object");
  97. }
  98. l = this.retrieveAvailable();
  99. } catch (NoSuchElementException e) {
  100. if (this.logger.isErrorEnabled()) {
  101. this.logger.error("could not remove ldap object from list", e);
  102. }
  103. throw new IllegalStateException("Pool is empty", e);
  104. }
  105. } else if (this.active.size() < this.poolConfig.getMaxPoolSize()) {
  106. if (this.logger.isTraceEnabled()) {
  107. this.logger.trace("pool can grow, attempt to create ldap object");
  108. }
  109. create = true;
  110. } else {
  111. if (this.logger.isTraceEnabled()) {
  112. this.logger.trace(
  113. "pool is full, block until ldap object " +
  114. "is available");
  115. }
  116. l = this.blockAvailable();
  117. }
  118. } finally {
  119. this.poolLock.unlock();
  120. }
  121. if (create) {
  122. // previous block determined a creation should occur
  123. // block here until create occurs without locking the whole pool
  124. // if the pool is already maxed or creates are failing,
  125. // block until an object is available
  126. this.checkOutLock.lock();
  127. try {
  128. boolean b = true;
  129. this.poolLock.lock();
  130. try {
  131. if (
  132. this.available.size() + this.active.size() ==
  133. this.poolConfig.getMaxPoolSize()) {
  134. b = false;
  135. }
  136. } finally {
  137. this.poolLock.unlock();
  138. }
  139. if (b) {
  140. l = this.createActive();
  141. if (this.logger.isTraceEnabled()) {
  142. this.logger.trace("created new active ldap object: " + l);
  143. }
  144. }
  145. } finally {
  146. this.checkOutLock.unlock();
  147. }
  148. if (l == null) {
  149. if (this.available.size() == 0 && this.active.size() == 0) {
  150. if (this.logger.isErrorEnabled()) {
  151. this.logger.error("Could not service check out request");
  152. }
  153. throw new LdapPoolExhaustedException(
  154. "Pool is empty and object creation failed");
  155. }
  156. if (this.logger.isDebugEnabled()) {
  157. this.logger.debug(
  158. "create failed, block until ldap object " +
  159. "is available");
  160. }
  161. l = this.blockAvailable();
  162. }
  163. }
  164. if (l != null) {
  165. this.activateAndValidate(l);
  166. } else {
  167. if (this.logger.isErrorEnabled()) {
  168. this.logger.error("Could not service check out request");
  169. }
  170. throw new LdapPoolExhaustedException(
  171. "Pool is empty and object creation failed");
  172. }
  173. return l;
  174. }
  175. /**
  176. * This attempts to retrieve an ldap object from the available queue.
  177. *
  178. * @return ldap object from the pool
  179. *
  180. * @throws NoSuchElementException if the available queue is empty
  181. */
  182. protected Ldap retrieveAvailable()
  183. {
  184. Ldap l = null;
  185. if (this.logger.isTraceEnabled()) {
  186. this.logger.trace(
  187. "waiting on pool lock for retrieve available " +
  188. this.poolLock.getQueueLength());
  189. }
  190. this.poolLock.lock();
  191. try {
  192. final PooledLdap<Ldap> pl = this.available.remove();
  193. this.active.add(new PooledLdap<Ldap>(pl.getLdap()));
  194. l = pl.getLdap();
  195. if (this.logger.isTraceEnabled()) {
  196. this.logger.trace("retrieved available ldap object: " + l);
  197. }
  198. } finally {
  199. this.poolLock.unlock();
  200. }
  201. return l;
  202. }
  203. /**
  204. * This blocks until an ldap object can be aquired.
  205. *
  206. * @return ldap object from the pool
  207. *
  208. * @throws LdapPoolException if this method fails
  209. * @throws BlockingTimeoutException if this pool is configured with a block
  210. * time and it occurs
  211. * @throws PoolInterruptedException if the current thread is interrupted
  212. */
  213. protected Ldap blockAvailable()
  214. throws LdapPoolException
  215. {
  216. Ldap l = null;
  217. if (this.logger.isTraceEnabled()) {
  218. this.logger.trace(
  219. "waiting on pool lock for block available " +
  220. this.poolLock.getQueueLength());
  221. }
  222. this.poolLock.lock();
  223. try {
  224. while (l == null) {
  225. if (this.logger.isTraceEnabled()) {
  226. this.logger.trace("available pool is empty, waiting...");
  227. }
  228. if (this.blockWaitTime > 0) {
  229. if (
  230. !this.poolNotEmpty.await(
  231. this.blockWaitTime,
  232. TimeUnit.MILLISECONDS)) {
  233. if (this.logger.isDebugEnabled()) {
  234. this.logger.debug("block time exceeded, throwing exception");
  235. }
  236. throw new BlockingTimeoutException("Block time exceeded");
  237. }
  238. } else {
  239. this.poolNotEmpty.await();
  240. }
  241. if (this.logger.isTraceEnabled()) {
  242. this.logger.trace("notified to continue...");
  243. }
  244. try {
  245. l = this.retrieveAvailable();
  246. } catch (NoSuchElementException e) {
  247. if (this.logger.isTraceEnabled()) {
  248. this.logger.trace("notified to continue but pool was empty");
  249. }
  250. }
  251. }
  252. } catch (InterruptedException e) {
  253. if (this.logger.isErrorEnabled()) {
  254. this.logger.error("waiting for available object interrupted", e);
  255. }
  256. throw new PoolInterruptedException(
  257. "Interrupted while waiting for an available object",
  258. e);
  259. } finally {
  260. this.poolLock.unlock();
  261. }
  262. return l;
  263. }
  264. /** {@inheritDoc} */
  265. public void checkIn(final Ldap l)
  266. {
  267. final boolean valid = this.validateAndPassivate(l);
  268. final PooledLdap<Ldap> pl = new PooledLdap<Ldap>(l);
  269. if (this.logger.isTraceEnabled()) {
  270. this.logger.trace(
  271. "waiting on pool lock for check in " + this.poolLock.getQueueLength());
  272. }
  273. this.poolLock.lock();
  274. try {
  275. if (this.active.remove(pl)) {
  276. if (valid) {
  277. this.available.add(pl);
  278. if (this.logger.isTraceEnabled()) {
  279. this.logger.trace("returned active ldap object: " + l);
  280. }
  281. this.poolNotEmpty.signal();
  282. }
  283. } else if (this.available.contains(pl)) {
  284. if (this.logger.isWarnEnabled()) {
  285. this.logger.warn("returned available ldap object: " + l);
  286. }
  287. } else {
  288. if (this.logger.isWarnEnabled()) {
  289. this.logger.warn("attempt to return unknown ldap object: " + l);
  290. }
  291. }
  292. } finally {
  293. this.poolLock.unlock();
  294. }
  295. }
  296. }