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

org.fabric3.threadpool.RuntimeThreadPoolExecutor Maven / Gradle / Ivy

The newest version!
/*
* Fabric3
* Copyright (c) 2009-2013 Metaform Systems
*
* Fabric3 is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as
* published by the Free Software Foundation, either version 3 of
* the License, or (at your option) any later version, with the
* following exception:
*
* Linking this software statically or dynamically with other
* modules is making a combined work based on this software.
* Thus, the terms and conditions of the GNU General Public
* License cover the whole combination.
*
* As a special exception, the copyright holders of this software
* give you permission to link this software with independent
* modules to produce an executable, regardless of the license
* terms of these independent modules, and to copy and distribute
* the resulting executable under terms of your choice, provided
* that you also meet, for each linked independent module, the
* terms and conditions of the license of that module. An
* independent module is a module which is not derived from or
* based on this software. If you modify this software, you may
* extend this exception to your version of the software, but
* you are not obligated to do so. If you do not wish to do so,
* delete this exception statement from your version.
*
* Fabric3 is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU General Public License for more details.
*
* You should have received a copy of the
* GNU General Public License along with Fabric3.
* If not, see .
*/
package org.fabric3.threadpool;

import java.util.List;
import java.util.concurrent.AbstractExecutorService;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.RejectedExecutionHandler;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;

import org.fabric3.api.annotation.management.Management;
import org.fabric3.api.annotation.management.ManagementOperation;
import org.fabric3.api.annotation.monitor.Monitor;
import org.fabric3.spi.threadpool.ExecutionContext;
import org.fabric3.spi.threadpool.ExecutionContextTunnel;
import org.fabric3.spi.threadpool.LongRunnable;
import org.oasisopen.sca.annotation.Destroy;
import org.oasisopen.sca.annotation.EagerInit;
import org.oasisopen.sca.annotation.Init;
import org.oasisopen.sca.annotation.Property;

/**
 * Processes work using a delegate {@link ThreadPoolExecutor}. This executor records processing statistics as well as monitors for stalled threads.
 * When a stalled thread is encountered (i.e. when the processing time for a runnable has exceeded a threshold), an event is sent to the monitor.
 * 

* The default configuration uses a bounded queue to accept work. If the queue size is exceeded, work will be rejected. This allows the runtime to * degrade gracefully under load by pushing requests back to the client and avoid out-of-memory conditions. */ @EagerInit @Management(name = "RuntimeThreadPoolExecutor", path = "/runtime/threadpool", group = "kernel", description = "Manages the runtime thread pool") public class RuntimeThreadPoolExecutor extends AbstractExecutorService { private int coreSize = 100; private long keepAliveTime = 60000; private boolean allowCoreThreadTimeOut = true; private int maximumSize = 100; private int queueSize = 10000; private int stallThreshold = 600000; private boolean checkStalledThreads = true; private long stallCheckPeriod = 60000; private RejectedExecutionHandler rejectedExecutionHandler = new ThreadPoolExecutor.CallerRunsPolicy(); private ThreadPoolExecutor delegate; private LinkedBlockingQueue queue; private StalledThreadMonitor stalledMonitor; private ExecutorMonitor monitor; private boolean statisticsOff = true; // queue of in-flight work private ConcurrentLinkedQueue inFlight = new ConcurrentLinkedQueue(); // statistics private AtomicLong totalExecutionTime = new AtomicLong(); private AtomicLong completedWorkCount = new AtomicLong(); /** * Sets the number of threads always available to service the executor queue. * * @param size the number of threads. */ @Property(required = false) public void setCoreSize(int size) { if (size < 0) { throw new IllegalArgumentException("Core pool size must be greater than or equal to 0"); } this.coreSize = size; } /** * Sets the maximum number of threads to service the executor queue. * * @param size the number of threads. */ @Property(required = false) public void setMaximumSize(int size) { if (size < 0) { throw new IllegalArgumentException("The MaximumSize pool size must be greater than or equal to 0"); } this.maximumSize = size; } /** * Sets the maximum number of work items that can be queued before the executor rejects additional work. * * @param size the maximum number of work items */ @Property(required = false) public void setQueueSize(int size) { this.queueSize = size; } /** * Sets the period between checks for stalled threads. * * @param period the period between checks for stalled threads. */ @Property(required = false) public void setStallCheckPeriod(long period) { this.stallCheckPeriod = period; } @Property(required = false) public void setRejectedExecutionHandler(String handler) { if ("abort".equals(handler)) { this.rejectedExecutionHandler = new ThreadPoolExecutor.AbortPolicy(); } else if ("discard".equals(handler)) { this.rejectedExecutionHandler = new ThreadPoolExecutor.DiscardPolicy(); } else if ("discard.oldest".equals(handler)) { this.rejectedExecutionHandler = new ThreadPoolExecutor.DiscardOldestPolicy(); } else if (!"caller.runs".equals(handler)) { monitor.error("Invalid rejected execution handler configuration - setting to caller.runs: " + handler); } } /** * Sets the time a thread can be processing work before it is considered stalled. The default is ten minutes. * * @param stallThreshold the time a thread can be processing work before it is considered stalled */ @Property(required = false) @ManagementOperation(description = "The time a thread can be processing work before it is considered stalled in milliseconds") public void setStallThreshold(int stallThreshold) { this.stallThreshold = stallThreshold; } @ManagementOperation(description = "Thread keep alive time in milliseconds") public long getKeepAliveTime() { return keepAliveTime; } @ManagementOperation(description = "Thread keep alive time in milliseconds") @Property(required = false) public void setKeepAliveTime(long keepAliveTime) { if (keepAliveTime < 0) { throw new IllegalArgumentException("Keep alive time must be greater than or equal to 0"); } this.keepAliveTime = keepAliveTime; } @ManagementOperation(description = "True if the thread pool expires core threads") public boolean isAllowCoreThreadTimeOut() { return allowCoreThreadTimeOut; } @ManagementOperation(description = "True if the thread pool expires core threads") @Property(required = false) public void setAllowCoreThreadTimeOut(boolean allowCoreThreadTimeOut) { this.allowCoreThreadTimeOut = allowCoreThreadTimeOut; } @ManagementOperation(description = "True warnings should be issued for stalled threads") public boolean isCheckStalledThreads() { return checkStalledThreads; } @Property(required = false) public void setCheckStalledThreads(boolean checkStalledThreads) { this.checkStalledThreads = checkStalledThreads; } @Property(required = false) public void setStatisticsOff(boolean statisticsOff) { this.statisticsOff = statisticsOff; } @ManagementOperation(description = "The time a thread can be processing work before it is considered stalled in milliseconds") public int getStallThreshold() { return stallThreshold; } @ManagementOperation(description = "Returns the approximate number of threads actively executing tasks") public int getActiveCount() { return delegate.getActiveCount(); } @ManagementOperation(description = "The maximum thread pool size") public int getMaximumPoolSize() { return delegate.getMaximumPoolSize(); } @ManagementOperation(description = "The maximum thread pool size") public void setMaximumPoolSize(int size) { delegate.setMaximumPoolSize(size); } @ManagementOperation(description = "The core thread pool size") public int getCorePoolSize() { return delegate.getCorePoolSize(); } @ManagementOperation(description = "The core thread pool size") public void setCorePoolSize(int size) { delegate.setCorePoolSize(size); } @ManagementOperation(description = "Returns the largest size the thread pool reached") public int getLargestPoolSize() { return delegate.getLargestPoolSize(); } @ManagementOperation(description = "Returns the remaining capacity the receive queue has before additional work will be rejected") public int getRemainingCapacity() { return queue.remainingCapacity(); } @ManagementOperation(description = "Returns the total time the thread pool has spent executing requests in milliseconds") public long getTotalExecutionTime() { return totalExecutionTime.get(); } @ManagementOperation(description = "Returns the total number of work items processed by the thread pool") public long getCompletedWorkCount() { return completedWorkCount.get(); } @ManagementOperation(description = "Returns the average elapsed time to process a work request in milliseconds") public double getMeanExecutionTime() { long count = completedWorkCount.get(); long totalTime = totalExecutionTime.get(); return count == 0 ? 0 : totalTime / count; } @ManagementOperation(description = "Returns the longest elapsed time for a currently running work request in milliseconds") public long getLongestRunning() { Runnable runnable = inFlight.peek(); if (runnable == null || !(runnable instanceof RunnableWrapper)) { // no work or statistics turned off return -1; } RunnableWrapper wrapper = (RunnableWrapper) runnable; return System.currentTimeMillis() - wrapper.start; } @ManagementOperation public int getCount() { return inFlight.size(); } public RuntimeThreadPoolExecutor(@Monitor ExecutorMonitor monitor) { this.monitor = monitor; } @Init public void init() { if (maximumSize < coreSize) { throw new IllegalArgumentException("Maximum pool size cannot be less than core pool size"); } if (queueSize > 0) { // create a bounded queue to accept work queue = new LinkedBlockingQueue(queueSize); } else { // create an unbounded queue to accept work queue = new LinkedBlockingQueue(); } RuntimeThreadFactory factory = new RuntimeThreadFactory(monitor); delegate = new ThreadPoolExecutor(coreSize, maximumSize, Long.MAX_VALUE, TimeUnit.SECONDS, queue, factory); delegate.setKeepAliveTime(keepAliveTime, TimeUnit.MILLISECONDS); delegate.allowCoreThreadTimeOut(allowCoreThreadTimeOut); if (checkStalledThreads && !statisticsOff) { stalledMonitor = new StalledThreadMonitor(); delegate.execute(stalledMonitor); } // set rejection strategy delegate.setRejectedExecutionHandler(rejectedExecutionHandler); } @Destroy public void stop() { if (stalledMonitor != null) { stalledMonitor.stop(); } delegate.shutdown(); } public void execute(Runnable runnable) { if (statisticsOff) { delegate.execute(runnable); } else { Runnable wrapper = new RunnableWrapper(runnable); delegate.execute(wrapper); } } public void shutdown() { delegate.shutdown(); } public List shutdownNow() { return delegate.shutdownNow(); } public boolean isShutdown() { return delegate.isShutdown(); } public boolean isTerminated() { return delegate.isTerminated(); } public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { return false; } /** * Wraps submitted work to record processing statistics. */ private class RunnableWrapper implements Runnable, ExecutionContext { private Runnable delegate; private boolean autoStart; private Thread currentThread; private long start = -1; private RunnableWrapper(Runnable delegate) { this.delegate = delegate; autoStart = !(delegate instanceof LongRunnable); } public void run() { ExecutionContext old = ExecutionContextTunnel.setThreadExecutionContext(this); try { if (autoStart) { start(); delegate.run(); stop(); } else { delegate.run(); } } finally { ExecutionContextTunnel.setThreadExecutionContext(old); clear(); } } public void start() { if (currentThread != null) { // already started, ignore return; } currentThread = Thread.currentThread(); inFlight.add(this); start = System.currentTimeMillis(); } public void clear() { currentThread = null; inFlight.remove(this); } public void stop() { long elapsed = System.currentTimeMillis() - start; totalExecutionTime.addAndGet(elapsed); completedWorkCount.incrementAndGet(); } } /** * Periodically scans in-flight work for threads that have exceeded a processing time threshold. */ private class StalledThreadMonitor implements Runnable { private AtomicBoolean active = new AtomicBoolean(true); public void run() { while (active.get()) { try { Thread.sleep(stallCheckPeriod); } catch (InterruptedException e) { Thread.interrupted(); continue; } // iterator never throws ConcurrentModificationException and can therefore be used to safely traverse the in-flight work queue for (Runnable runnable : inFlight) { if (!(runnable instanceof RunnableWrapper)) { continue; } RunnableWrapper wrapper = (RunnableWrapper) runnable; long elapsed = System.currentTimeMillis() - wrapper.start; if (elapsed >= stallThreshold) { Thread thread = wrapper.currentThread; if (thread != null) { StackTraceElement[] trace = thread.getStackTrace(); StringBuilder builder = new StringBuilder(); for (StackTraceElement element : trace) { builder.append("\tat ").append(element).append("\n"); } monitor.stalledThread(thread.getName(), elapsed, builder.toString()); } } } } } public void stop() { active.set(false); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy