com.wl4g.components.common.task.SafeScheduledTaskPoolExecutor Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2017 ~ 2025 the original author or authors.
*
* Licensed 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 com.wl4g.components.common.task;
import static com.wl4g.components.common.lang.Assert2.isTrue;
import static com.wl4g.components.common.lang.Assert2.notNull;
import static com.wl4g.components.common.lang.Assert2.notNullOf;
import static com.wl4g.components.common.log.SmartLoggerFactory.getLogger;
import static com.wl4g.components.common.reflect.ReflectionUtils2.*;
import static java.lang.Integer.MAX_VALUE;
import static java.lang.String.format;
import static java.lang.System.nanoTime;
import static java.util.Collections.emptyList;
import static java.util.Objects.nonNull;
import static java.util.concurrent.ThreadLocalRandom.current;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.NANOSECONDS;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.Delayed;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.RejectedExecutionHandler;
import java.util.concurrent.RunnableScheduledFuture;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicLong;
import org.slf4j.Logger;
import com.wl4g.components.common.collection.CollectionUtils2;
/**
* An enhanced security and flexible scheduling executor.
* As the default {@link java.util.concurrent.ScheduledThreadPoolExecutor} and
* {@link org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler} of
* JDK do not limit the maximum task waiting queue, the problem of oom may
* occur, which is designed to solve this problem.
*
* @author Wangl.sir
* @version v1.0 2020年1月18日
* @since
* @see {@link org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler}
* @see Resolution
* ScheduledThreadPoolExecutor for retry task OOM
*/
public class SafeScheduledTaskPoolExecutor extends ScheduledThreadPoolExecutor {
final protected Logger log = getLogger(getClass());
/**
* Maximum allowed waiting execution queue size.
*/
final private int acceptQueue;
/**
* {@link RejectedExecutionHandler}
*/
final private RejectedExecutionHandler rejectHandler;
public SafeScheduledTaskPoolExecutor(int coreMaximumPoolSize, long keepAliveTimeMs, ThreadFactory threadFactory,
int acceptQueue, RejectedExecutionHandler rejectHandler) {
super(coreMaximumPoolSize, threadFactory, rejectHandler);
isTrue(acceptQueue > 0, "acceptQueue must be greater than 0");
notNullOf(rejectHandler, "rejectHandler");
setMaximumPoolSize(coreMaximumPoolSize); // corePoolSize==maximumPoolSize
setKeepAliveTime(keepAliveTimeMs, MILLISECONDS);
this.acceptQueue = acceptQueue;
this.rejectHandler = rejectHandler;
}
/**
* The {@link #invokeAll(Collection, long, TimeUnit)} or
* {@link #invokeAny(Collection, long, TimeUnit)} related methods also call
* {@link #execute(Runnable)} in the end. For details, see:
*
* @see {@link java.util.concurrent.AbstractExecutorService#doInvokeAny()#176}
* @see {@link java.util.concurrent.ExecutorCompletionService#submit()}
*/
@Override
public void execute(Runnable command) {
if (checkRejectedQueueLimit(command))
return;
super.execute(command);
}
@Override
public Future> submit(Runnable task) {
if (checkRejectedQueueLimit(task))
return null;
return super.submit(task);
}
@Override
public Future submit(Runnable task, T result) {
if (checkRejectedQueueLimit(task))
return null;
return super.submit(task, result);
}
/**
* Submitted job wait for completed.
*
* @param jobs
* @param timeoutMs
* @throws IllegalStateException
*/
public void submitForComplete(List jobs, long timeoutMs) throws IllegalStateException {
submitForComplete(jobs, (ex, completed, uncompleted) -> {
if (nonNull(ex)) {
throw ex;
}
}, timeoutMs);
}
/**
* Submitted job wait for completed.
*
* @param jobs
* @param listener
* @param timeoutMs
* @throws IllegalStateException
*/
public void submitForComplete(List jobs, CompleteTaskListener listener, long timeoutMs)
throws IllegalStateException {
if (!CollectionUtils2.isEmpty(jobs)) {
int total = jobs.size();
// Future jobs.
Map, Runnable> futures = new HashMap, Runnable>(total);
try {
CountDownLatch latch = new CountDownLatch(total);
// Submit job.
jobs.stream().forEach(job -> futures.put(submit(new FutureDoneTask(latch, job)), job));
if (!latch.await(timeoutMs, MILLISECONDS)) { // Timeout?
Iterator, Runnable>> it = futures.entrySet().iterator();
while (it.hasNext()) {
Entry, Runnable> entry = it.next();
if (!entry.getKey().isCancelled() && !entry.getKey().isDone()) {
entry.getKey().cancel(true);
} else {
it.remove(); // Cleanup cancelled or isDone
}
}
TimeoutException ex = new TimeoutException(
format("Failed to job execution timeout, %s -> completed(%s)/total(%s)",
jobs.get(0).getClass().getName(), (total - latch.getCount()), total));
listener.onComplete(ex, (total - latch.getCount()), futures.values());
} else {
listener.onComplete(null, total, emptyList());
}
} catch (Exception e) {
throw new IllegalStateException(e);
}
}
}
@Override
public ScheduledFuture> schedule(Runnable command, long delay, TimeUnit unit) {
if (checkRejectedQueueLimit(command))
return null;
return super.schedule(command, delay, unit);
}
@Override
public ScheduledFuture schedule(Callable callable, long delay, TimeUnit unit) {
if (checkRejectedQueueLimit(callable))
return null;
return super.schedule(callable, delay, unit);
}
/**
* Random interval scheduling based on dynamic schedule.
*
* @see {@link java.util.concurrent.ScheduledThreadPoolExecutor#scheduleAtFixedRate(Runnable, long, long, TimeUnit)}
*
* @param runnable
* @param initialDelay
* @param minDelay
* @param maxDelay
* @param unit
* @return
*/
public ScheduledFuture> scheduleAtRandomRate(Runnable runnable, long initialDelay, long minDelay, long maxDelay,
TimeUnit unit) {
return scheduleAtFixedRate(new RandomScheduleRunnable(unit.toMillis(minDelay), unit.toMillis(maxDelay), runnable),
unit.toMillis(initialDelay), MAX_VALUE, MILLISECONDS);
}
/**
* Random interval scheduling based on fixed schedule.
*
* @see {@link java.util.concurrent.ScheduledThreadPoolExecutor#scheduleWithFixedDelay(Runnable, long, long, TimeUnit)}
*
* @param runnable
* @param initialDelay
* @param minDelay
* @param maxDelay
* @param unit
* @return
*/
public ScheduledFuture> scheduleWithRandomDelay(Runnable runnable, long initialDelay, long minDelay, long maxDelay,
TimeUnit unit) {
return scheduleWithFixedDelay(new RandomScheduleRunnable(unit.toMillis(minDelay), unit.toMillis(maxDelay), runnable),
unit.toMillis(initialDelay), MAX_VALUE, MILLISECONDS);
}
@Override
public ScheduledFuture> scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit) {
if (checkRejectedQueueLimit(command))
return null;
return super.scheduleAtFixedRate(command, initialDelay, period, unit);
}
@Override
public ScheduledFuture> scheduleWithFixedDelay(Runnable command, long initialDelay, long delay, TimeUnit unit) {
if (checkRejectedQueueLimit(command))
return null;
return super.scheduleWithFixedDelay(command, initialDelay, delay, unit);
}
/**
* @see {@link java.util.concurrent.ScheduledThreadPoolExecutor.ScheduledFutureTask}
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
@Override
protected RunnableScheduledFuture decorateTask(Runnable runnable, RunnableScheduledFuture task) {
if (runnable instanceof RandomScheduleRunnable) {
try {
return new CustomScheduledFutureTask((Callable) callableField.get(task), (long) timeField.get(task), this) {
@Override
public long getPeriod() {
return ((RandomScheduleRunnable) runnable).nextDelay();
}
};
} catch (Exception e) {
throw new IllegalStateException(e);
}
}
return task;
}
/**
* @see {@link java.util.concurrent.ScheduledThreadPoolExecutor.ScheduledFutureTask}
*/
@Override
protected RunnableScheduledFuture decorateTask(Callable callable, RunnableScheduledFuture task) {
return decorateTask(EMPTY_RUNNABLE, task);
}
/**
* Check whether the entry queue is rejected
*
* @param command
* @return
*/
private boolean checkRejectedQueueLimit(Callable> command) {
if (getQueue().size() > acceptQueue) {
rejectHandler.rejectedExecution(() -> {
try {
command.call();
} catch (Exception e) {
throw new IllegalStateException();
}
}, this);
return true;
}
return false;
}
/**
* Check whether the entry queue is rejected
*
* @param command
* @return
*/
private boolean checkRejectedQueueLimit(Runnable command) {
if (getQueue().size() > acceptQueue) {
rejectHandler.rejectedExecution(command, this);
// throw new RejectedExecutionException("Rejected execution of " + r
// + " on " + executor, executor.isShutdown());
return true;
}
return false;
}
/**
* @see {@link ScheduledFutureTask}
*
* @see {@link ScheduledThreadPoolExecutor.ScheduledFutureTask}
* @param
* @author Wangl.sir <[email protected], [email protected]>
* @version 2020年1月20日 v1.0.0
* @see
*/
private class CustomScheduledFutureTask extends FutureTask implements RunnableScheduledFuture {
/**
* {@link ScheduledThreadPoolExecutor} instance object.
*/
private ScheduledThreadPoolExecutor executor;
/** Sequence number to break ties FIFO */
private final long sequenceNumber;
/** The time the task is enabled to execute in nanoTime units */
private long time;
/** The actual task to be re-enqueued by reExecutePeriodic */
RunnableScheduledFuture outerTask = this;
/**
* Index into delay queue, to support faster cancellation.
*/
int heapIndex;
/**
* Creates a one-shot action with given nanoTime-based trigger time.
*/
CustomScheduledFutureTask(Callable callable, long ns, ScheduledThreadPoolExecutor executor) {
super(callable);
this.time = ns;
this.executor = executor;
this.sequenceNumber = sequencer.getAndIncrement();
}
public long getDelay(TimeUnit unit) {
return unit.convert(time - nanoTime(), NANOSECONDS);
}
public int compareTo(Delayed other) {
if (other == this) // compare zero if same object
return 0;
if (other instanceof CustomScheduledFutureTask) {
CustomScheduledFutureTask> x = (CustomScheduledFutureTask>) other;
long diff = time - x.time;
if (diff < 0)
return -1;
else if (diff > 0)
return 1;
else if (sequenceNumber < x.sequenceNumber)
return -1;
else
return 1;
}
long diff = getDelay(NANOSECONDS) - other.getDelay(NANOSECONDS);
return (diff < 0) ? -1 : (diff > 0) ? 1 : 0;
}
/**
* Returns {@code true} if this is a periodic (not a one-shot) action.
*
* @return {@code true} if periodic
*/
public boolean isPeriodic() {
return getPeriod() != 0;
}
/**
* Period in nanoseconds for repeating tasks. A positive value indicates
* fixed-rate execution. A negative value indicates fixed-delay
* execution. A value of 0 indicates a non-repeating task.
*/
public long getPeriod() {
return 0;
}
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
boolean cancelled = super.cancel(mayInterruptIfRunning);
if (cancelled && getRemoveOnCancelPolicy() && heapIndex >= 0)
remove(this);
return cancelled;
}
/**
* Overrides FutureTask version so as to reset/requeue if periodic.
*/
@Override
public void run() {
boolean periodic = isPeriodic();
if (!canRunInCurrentRunState(periodic))
cancel(false);
else if (!periodic)
CustomScheduledFutureTask.super.run();
else if (CustomScheduledFutureTask.super.runAndReset()) {
setNextRunTime();
// reExecutePeriodic(outerTask);
try {
reExecutePeriodicMethod.invoke(executor, outerTask);
} catch (Exception e) {
throw new IllegalStateException(e);
}
}
}
/**
* Sets the next time to run for a periodic task.
*/
private void setNextRunTime() {
long p = getPeriod();
if (p > 0)
time += p;
else
time = triggerTime(-p);
}
/**
* Returns the trigger time of a delayed action.
*/
private long triggerTime(long delay) {
return nanoTime() + ((delay < (Long.MAX_VALUE >> 1)) ? delay : overflowFree(delay));
}
/**
* Constrains the values of all delays in the queue to be within
* Long.MAX_VALUE of each other, to avoid overflow in compareTo. This
* may occur if a task is eligible to be dequeued, but has not yet been,
* while some other task is added with a delay of Long.MAX_VALUE.
*/
private long overflowFree(long delay) {
Delayed head = (Delayed) getQueue().peek();
if (head != null) {
long headDelay = head.getDelay(NANOSECONDS);
if (headDelay < 0 && (delay - headDelay < 0))
delay = Long.MAX_VALUE + headDelay;
}
return delay;
}
/**
* Returns true if can run a task given current run state and
* run-after-shutdown parameters.
*
* @param periodic
* true if this task periodic, false if delayed
*/
private boolean canRunInCurrentRunState(boolean periodic) {
// return super.canRunInCurrentRunState(periodic);
try {
return (boolean) canRunInCurrentRunStateMethod.invoke(executor, periodic);
} catch (Exception e) {
throw new IllegalStateException(e);
}
}
}
/**
* Future done runnable wrapper.
*
* @author Wangl.sir
* @version v1.0 2019年10月17日
* @since
*/
private class FutureDoneTask implements Runnable {
/** {@link CountDownLatch} */
final private CountDownLatch latch;
/** Real runner job. */
final private Runnable job;
public FutureDoneTask(CountDownLatch latch, Runnable job) {
notNull(latch, "Job runable latch must not be null.");
notNull(job, "Job runable must not be null.");
this.latch = latch;
this.job = job;
}
@Override
public void run() {
try {
job.run();
} catch (Exception e) {
log.error("Execution failure task", e);
} finally {
latch.countDown();
}
}
}
/**
* {@link RandomScheduleRunnable}
*
* @author Wangl.sir <[email protected], [email protected]>
* @version 2020年1月20日 v1.0.0
* @see
*/
public static class RandomScheduleRunnable implements Runnable {
/**
* Random min delay ms.
*/
final private long minDelayMs;
/**
* Random max delay ms.
*/
final private long maxDelayMs;
/**
* Runnable
*/
final private Runnable runnable;
public RandomScheduleRunnable(long minDelayMs, long maxDelayMs, Runnable runnable) {
notNullOf(minDelayMs, "minDelayMs");
notNullOf(maxDelayMs, "maxDelayMs");
notNullOf(runnable, "runnable");
this.minDelayMs = minDelayMs;
this.maxDelayMs = maxDelayMs;
this.runnable = runnable;
}
public long nextDelay() {
return MILLISECONDS.toNanos(current().nextLong(minDelayMs, maxDelayMs));
}
@Override
public void run() {
runnable.run();
}
}
/**
* Empty runnable.
*/
final private static Runnable EMPTY_RUNNABLE = () -> {
};
/**
* Sequence number to break scheduling ties, and in turn to guarantee FIFO
* order among tied entries.
*/
final private static AtomicLong sequencer = new AtomicLong();
/**
* {@link java.util.concurrent.ScheduledThreadPoolExecutor#reExecutePeriodic(RunnableScheduledFuture)}
*/
final private static Method reExecutePeriodicMethod;
/**
* {@link java.util.concurrent.ScheduledThreadPoolExecutor#canRunInCurrentRunState(boolean)}
*/
final private static Method canRunInCurrentRunStateMethod;
/**
* {@link java.util.concurrent.ScheduledThreadPoolExecutor.ScheduledFutureTask#callable}
*/
final private static Field callableField;
/**
* {@link java.util.concurrent.ScheduledThreadPoolExecutor.ScheduledFutureTask#time}
*/
final private static Field timeField;
static {
try {
Class> scheduleFutureTaskClass = Class
.forName("java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask");
callableField = findField(scheduleFutureTaskClass, "callable");
makeAccessible(callableField);
timeField = findField(scheduleFutureTaskClass, "time");
makeAccessible(timeField);
reExecutePeriodicMethod = ScheduledThreadPoolExecutor.class.getDeclaredMethod("reExecutePeriodic",
RunnableScheduledFuture.class);
makeAccessible(reExecutePeriodicMethod);
canRunInCurrentRunStateMethod = ScheduledThreadPoolExecutor.class.getDeclaredMethod("canRunInCurrentRunState",
boolean.class);
makeAccessible(canRunInCurrentRunStateMethod);
} catch (Exception e) {
throw new IllegalStateException(e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy