/src/org/opencms/ugc/CmsUgcSessionQueue.java

http://github.com/alkacon/opencms-core · Java · 138 lines · 57 code · 17 blank · 64 comment · 8 complexity · 827a9f932cf34b46423f2f287b60747e MD5 · raw file

  1. /*
  2. * This library is part of OpenCms -
  3. * the Open Source Content Management System
  4. *
  5. * Copyright (c) Alkacon Software GmbH & Co. KG (http://www.alkacon.com)
  6. *
  7. * This library is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * This library is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * For further information about Alkacon Software, please see the
  18. * company website: http://www.alkacon.com
  19. *
  20. * For further information about OpenCms, please see the
  21. * project website: http://www.opencms.org
  22. *
  23. * You should have received a copy of the GNU Lesser General Public
  24. * License along with this library; if not, write to the Free Software
  25. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  26. */
  27. package org.opencms.ugc;
  28. /**
  29. * A queue used to enforce a certain wait time between requests for sessions for the same form configuration.<p>
  30. */
  31. public class CmsUgcSessionQueue {
  32. /** True if the session queue is enabled. */
  33. private boolean m_enabled;
  34. /** The wait time between multiple session requests for the same configuration. */
  35. private long m_interval;
  36. /** The maximum number of threads waiting in the queue for the same configuration. */
  37. private int m_maxLength;
  38. /** The time at which the next thread should be scheduled to run. */
  39. private long m_nextScheduleTime;
  40. /** The number of threads waiting on this queue. */
  41. private int m_waitCount;
  42. /**
  43. * Creates a new instance.<p>
  44. *
  45. * @param enabled true if the queue should be enabled
  46. * @param interval the wait time to use between multiple session requests
  47. * @param maxLength the maximum length of the queue
  48. */
  49. public CmsUgcSessionQueue(boolean enabled, long interval, int maxLength) {
  50. m_enabled = enabled;
  51. m_interval = interval;
  52. m_maxLength = maxLength;
  53. if (interval < 0) {
  54. throw new IllegalArgumentException("Queue wait time must not be < 0");
  55. }
  56. if (maxLength < 0) {
  57. throw new IllegalArgumentException("Queue maximum length must not be < 0");
  58. }
  59. }
  60. /**
  61. * Creates a session queue based on the given configuration.<p>
  62. *
  63. * @param config the configuration for which to create the session queue
  64. * @return the newly created session queue
  65. */
  66. public static CmsUgcSessionQueue createQueue(CmsUgcConfiguration config) {
  67. CmsUgcSessionQueue queue = new CmsUgcSessionQueue(
  68. config.needsQueue(),
  69. config.getQueueInterval().isPresent() ? config.getQueueInterval().get().longValue() : 0,
  70. config.getMaxQueueLength().isPresent() ? config.getMaxQueueLength().get().intValue() : Integer.MAX_VALUE);
  71. return queue;
  72. }
  73. /**
  74. * Updates the queue parameters from the configuration object.<p>
  75. *
  76. * @param config the form configuration
  77. */
  78. public synchronized void updateFromConfiguration(CmsUgcConfiguration config) {
  79. m_enabled = config.needsQueue();
  80. m_interval = config.getQueueInterval().isPresent() ? config.getQueueInterval().get().longValue() : 0;
  81. m_maxLength = config.getMaxQueueLength().isPresent()
  82. ? config.getMaxQueueLength().get().intValue()
  83. : Integer.MAX_VALUE;
  84. }
  85. /**
  86. * If there are currently any threads waiting on this queue, wait for the interval given on construction after the currenly last thread stops waiting.<p>
  87. *
  88. * @return false if the queue was too long to wait, true otherwise
  89. */
  90. public synchronized boolean waitForSlot() {
  91. if (!m_enabled) {
  92. return true;
  93. } else {
  94. long now = System.currentTimeMillis();
  95. long timeToWait = m_nextScheduleTime - now;
  96. if (timeToWait <= 0) {
  97. // This happens either if this method is called for the first time,
  98. // or if the wait interval for the last thread to enter the queue has
  99. // already fully elapsed
  100. m_nextScheduleTime = now + m_interval;
  101. return true;
  102. } else if (m_waitCount >= m_maxLength) {
  103. return false;
  104. } else {
  105. m_nextScheduleTime = m_nextScheduleTime + m_interval;
  106. m_waitCount += 1;
  107. try {
  108. // use wait here instead of sleep because it releases the object monitor
  109. // note that timeToWait can't be 0 here, because this case is handled in the first if() branch,
  110. // so wait() always returns after the given period
  111. wait(timeToWait);
  112. } catch (InterruptedException e) {
  113. // TODO Auto-generated catch block
  114. e.printStackTrace();
  115. }
  116. m_waitCount -= 1;
  117. return true;
  118. }
  119. }
  120. }
  121. }