/vt-ldap/tags/vt-ldap-3.3.1/src/main/java/edu/vt/middleware/ldap/pool/BlockingLdapPool.java

http://vt-middleware.googlecode.com/ · Java · 316 lines · 215 code · 24 blank · 77 comment · 43 complexity · a8d5569077b120a8e06d2aa76fd2d606 MD5 · raw file

  1. /*
  2. $Id: BlockingLdapPool.java 1330 2010-05-23 22:10:53Z 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: 1330 $
  9. Updated: $Date: 2010-05-24 00:10:53 +0200 (Mon, 24 May 2010) $
  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: 1330 $ $Date: 2010-05-24 00:10:53 +0200 (Mon, 24 May 2010) $
  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.logger.isDebugEnabled()) {
  150. this.logger.debug(
  151. "create failed, block until ldap object " +
  152. "is available");
  153. }
  154. l = this.blockAvailable();
  155. }
  156. }
  157. if (l != null) {
  158. this.activateAndValidate(l);
  159. } else {
  160. if (this.logger.isErrorEnabled()) {
  161. this.logger.error("Could not service check out request");
  162. }
  163. throw new LdapPoolExhaustedException(
  164. "Pool is empty and object creation failed");
  165. }
  166. return l;
  167. }
  168. /**
  169. * This attempts to retrieve an ldap object from the available queue.
  170. *
  171. * @return ldap object from the pool
  172. *
  173. * @throws NoSuchElementException if the available queue is empty
  174. */
  175. protected Ldap retrieveAvailable()
  176. {
  177. Ldap l = null;
  178. if (this.logger.isTraceEnabled()) {
  179. this.logger.trace(
  180. "waiting on pool lock for retrieve available " +
  181. this.poolLock.getQueueLength());
  182. }
  183. this.poolLock.lock();
  184. try {
  185. final PooledLdap<Ldap> pl = this.available.remove();
  186. this.active.add(new PooledLdap<Ldap>(pl.getLdap()));
  187. l = pl.getLdap();
  188. if (this.logger.isTraceEnabled()) {
  189. this.logger.trace("retrieved available ldap object: " + l);
  190. }
  191. } finally {
  192. this.poolLock.unlock();
  193. }
  194. return l;
  195. }
  196. /**
  197. * This blocks until an ldap object can be aquired.
  198. *
  199. * @return ldap object from the pool
  200. *
  201. * @throws LdapPoolException if this method fails
  202. * @throws BlockingTimeoutException if this pool is configured with a block
  203. * time and it occurs
  204. * @throws PoolInterruptedException if the current thread is interrupted
  205. */
  206. protected Ldap blockAvailable()
  207. throws LdapPoolException
  208. {
  209. Ldap l = null;
  210. if (this.logger.isTraceEnabled()) {
  211. this.logger.trace(
  212. "waiting on pool lock for block available " +
  213. this.poolLock.getQueueLength());
  214. }
  215. this.poolLock.lock();
  216. try {
  217. while (l == null) {
  218. if (this.logger.isTraceEnabled()) {
  219. this.logger.trace("available pool is empty, waiting...");
  220. }
  221. if (this.blockWaitTime > 0) {
  222. if (
  223. !this.poolNotEmpty.await(
  224. this.blockWaitTime,
  225. TimeUnit.MILLISECONDS)) {
  226. if (this.logger.isDebugEnabled()) {
  227. this.logger.debug("block time exceeded, throwing exception");
  228. }
  229. throw new BlockingTimeoutException("Block time exceeded");
  230. }
  231. } else {
  232. this.poolNotEmpty.await();
  233. }
  234. if (this.logger.isTraceEnabled()) {
  235. this.logger.trace("notified to continue...");
  236. }
  237. try {
  238. l = this.retrieveAvailable();
  239. } catch (NoSuchElementException e) {
  240. if (this.logger.isTraceEnabled()) {
  241. this.logger.trace("notified to continue but pool was empty");
  242. }
  243. }
  244. }
  245. } catch (InterruptedException e) {
  246. if (this.logger.isErrorEnabled()) {
  247. this.logger.error("waiting for available object interrupted", e);
  248. }
  249. throw new PoolInterruptedException(
  250. "Interrupted while waiting for an available object",
  251. e);
  252. } finally {
  253. this.poolLock.unlock();
  254. }
  255. return l;
  256. }
  257. /** {@inheritDoc} */
  258. public void checkIn(final Ldap l)
  259. {
  260. final boolean valid = this.validateAndPassivate(l);
  261. final PooledLdap<Ldap> pl = new PooledLdap<Ldap>(l);
  262. if (this.logger.isTraceEnabled()) {
  263. this.logger.trace(
  264. "waiting on pool lock for check in " + this.poolLock.getQueueLength());
  265. }
  266. this.poolLock.lock();
  267. try {
  268. if (this.active.remove(pl)) {
  269. if (valid) {
  270. this.available.add(pl);
  271. if (this.logger.isTraceEnabled()) {
  272. this.logger.trace("returned active ldap object: " + l);
  273. }
  274. this.poolNotEmpty.signal();
  275. }
  276. } else if (this.available.contains(pl)) {
  277. if (this.logger.isWarnEnabled()) {
  278. this.logger.warn("returned available ldap object: " + l);
  279. }
  280. } else {
  281. if (this.logger.isWarnEnabled()) {
  282. this.logger.warn("attempt to return unknown ldap object: " + l);
  283. }
  284. }
  285. } finally {
  286. this.poolLock.unlock();
  287. }
  288. }
  289. }