All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.activemq.artemis.rest.util.TimeoutTask Maven / Gradle / Ivy

There is a newer version: 2.25.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.activemq.artemis.rest.util;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

import org.apache.activemq.artemis.rest.ActiveMQRestLogger;

public class TimeoutTask implements Runnable {

   protected boolean running = true;
   protected int interval = 10;
   protected final Lock callbacksLock = new ReentrantLock();
   protected Map callbacks = new HashMap<>();
   protected final Lock pendingCallbacksLock = new ReentrantLock();
   protected Map pendingCallbacks = new HashMap<>();
   protected Thread thread;

   public TimeoutTask(int interval) {
      this.interval = interval;
   }

   public interface Callback {

      boolean testTimeout(String token, boolean autoShutdown);

      void shutdown(String token);
   }

   public synchronized void add(Callback callback, String token) {
      if (callbacksLock.tryLock()) {
         try {
            callbacks.put(token, callback);
         }
         finally {
            callbacksLock.unlock();
         }
      }
      else {
         pendingCallbacksLock.lock();
         try {
            pendingCallbacks.put(token, callback);
         }
         finally {
            pendingCallbacksLock.unlock();
         }
      }
   }

   public synchronized void remove(String token) {
      callbacksLock.lock();
      try {
         callbacks.remove(token);
      }
      finally {
         callbacksLock.unlock();
      }
   }

   public synchronized void stop() {
      running = false;
      thread.interrupt();
   }

   public synchronized int getInterval() {
      return interval;
   }

   public synchronized void setInterval(int interval) {
      this.interval = interval;
   }

   public void start() {
      thread = new Thread(this);
      thread.start();
   }

   @Override
   public void run() {
      while (running) {
         try {
            Thread.sleep(interval * 1000);
         }
         catch (InterruptedException e) {
            running = false;
            break;
         }

         // First, test all known callbacks for timeouts.
         // If the timeout is true, then move it to a separate map.
         Map expiredCallbacks = new HashMap<>();

         int liveConsumers = 0;
         int deadConsumers = 0;

         callbacksLock.lock();
         try {
            long startTime = System.currentTimeMillis();
            List tokens = new ArrayList<>(callbacks.size());
            for (String token : callbacks.keySet()) {
               tokens.add(token);
            }
            for (String token : tokens) {
               Callback callback = callbacks.get(token);
               if (callback.testTimeout(token, false)) {
                  deadConsumers += 1;
                  expiredCallbacks.put(token, callback);
                  callbacks.remove(token);
               }
               else {
                  liveConsumers += 1;
               }
            }
            ActiveMQRestLogger.LOGGER.debug("Finished testing callbacks for timeouts in " +
                                               (System.currentTimeMillis() - startTime) + "ms. " +
                                               "(Live: " + liveConsumers + ", Expired: " + deadConsumers + ")");

            // Next, move any pending callback additions to the main callbacks map.
            pendingCallbacksLock.lock();
            try {
               if (pendingCallbacks.size() > 0) {
                  ActiveMQRestLogger.LOGGER.debug("Found " + pendingCallbacks.size() + " callbacks to add.");
                  callbacks.putAll(pendingCallbacks);
                  pendingCallbacks.clear();
               }
            }
            finally {
               pendingCallbacksLock.unlock();
            }
         }
         finally {
            callbacksLock.unlock();
         }

         // Finally, freely shutdown all expired consumers.
         if (expiredCallbacks.size() > 0) {
            long startTime = System.currentTimeMillis();
            List tokens = new ArrayList<>(expiredCallbacks.size());
            for (String token : expiredCallbacks.keySet()) {
               tokens.add(token);
            }
            for (String token : tokens) {
               Callback expired = expiredCallbacks.get(token);
               expired.shutdown(token);
            }
         }
      }
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy