From 92e4d73dcb69266984f5a566fe8885cf3ab8f733 Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Fri, 13 Sep 2019 09:52:28 +1000 Subject: [PATCH 1/2] Issue #1036 Configure Scheduler Allows scheduler configuration Signed-off-by: Greg Wilkins --- jetty-server/src/main/config/etc/jetty.xml | 6 ++- .../src/main/config/modules/server.mod | 5 +++ .../thread/ScheduledExecutorScheduler.java | 39 +++++++++++++++---- 3 files changed, 42 insertions(+), 8 deletions(-) diff --git a/jetty-server/src/main/config/etc/jetty.xml b/jetty-server/src/main/config/etc/jetty.xml index 02a43bf2e56..a09928522af 100644 --- a/jetty-server/src/main/config/etc/jetty.xml +++ b/jetty-server/src/main/config/etc/jetty.xml @@ -15,7 +15,11 @@ - + + + + + diff --git a/jetty-server/src/main/config/modules/server.mod b/jetty-server/src/main/config/modules/server.mod index dd039a01fb1..55ad19b9bec 100644 --- a/jetty-server/src/main/config/modules/server.mod +++ b/jetty-server/src/main/config/modules/server.mod @@ -84,3 +84,8 @@ patch-module: servlet.api=lib/jetty-schemas-3.1.jar ## Dump the state of the Jetty server, components, and webapps before shutdown # jetty.server.dumpBeforeStop=false + +## Scheduler Configuration +# jetty.scheduler.name= +# jetty.scheduler.deamon=false +# jetty.scheduler.threads=-1 diff --git a/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java b/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java index a0c889d162c..90a629b5205 100644 --- a/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java +++ b/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java @@ -22,7 +22,10 @@ import java.io.IOException; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicInteger; +import org.eclipse.jetty.util.ProcessorUtils; +import org.eclipse.jetty.util.annotation.Name; import org.eclipse.jetty.util.component.AbstractLifeCycle; import org.eclipse.jetty.util.component.Dumpable; @@ -40,6 +43,8 @@ public class ScheduledExecutorScheduler extends AbstractLifeCycle implements Sch private final boolean daemon; private final ClassLoader classloader; private final ThreadGroup threadGroup; + private final int threads; + private final AtomicInteger count = new AtomicInteger(); private volatile ScheduledThreadPoolExecutor scheduler; private volatile Thread thread; @@ -50,28 +55,48 @@ public class ScheduledExecutorScheduler extends AbstractLifeCycle implements Sch public ScheduledExecutorScheduler(String name, boolean daemon) { - this(name, daemon, Thread.currentThread().getContextClassLoader()); + this(name, daemon, null); } - public ScheduledExecutorScheduler(String name, boolean daemon, ClassLoader threadFactoryClassLoader) + public ScheduledExecutorScheduler(@Name("name") String name, @Name("daemon") boolean daemon, @Name("threads") int threads) { - this(name, daemon, threadFactoryClassLoader, null); + this(name, daemon, null, null, threads); } - public ScheduledExecutorScheduler(String name, boolean daemon, ClassLoader threadFactoryClassLoader, ThreadGroup threadGroup) + public ScheduledExecutorScheduler(String name, boolean daemon, ClassLoader classLoader) + { + this(name, daemon, classLoader, null); + } + + public ScheduledExecutorScheduler(String name, boolean daemon, ClassLoader classLoader, ThreadGroup threadGroup) + { + this(name, daemon, classLoader, threadGroup, -1); + } + + /** + * @param name The name of the scheduler threads or null for automatic name + * @param daemon True if scheduler threads should be daemon + * @param classLoader The classloader to run the threads with or null to use the current thread context classloader + * @param threadGroup The threadgroup to use or null for no thread group + * @param threads The number of threads to pass to the the core {@link ScheduledThreadPoolExecutor} or -1 for a + * heuristic determined number of threads. + */ + public ScheduledExecutorScheduler(@Name("name") String name, @Name("daemon") boolean daemon, @Name("classLoader") ClassLoader classLoader, @Name("threadGroup") ThreadGroup threadGroup, @Name("threads") int threads) { this.name = name == null ? "Scheduler-" + hashCode() : name; this.daemon = daemon; - this.classloader = threadFactoryClassLoader == null ? Thread.currentThread().getContextClassLoader() : threadFactoryClassLoader; + this.classloader = classLoader == null ? Thread.currentThread().getContextClassLoader() : classLoader; this.threadGroup = threadGroup; + this.threads = threads; } @Override protected void doStart() throws Exception { - scheduler = new ScheduledThreadPoolExecutor(1, r -> + int size = threads > 0 ? threads : Math.max(1, ProcessorUtils.availableProcessors() / 4); + scheduler = new ScheduledThreadPoolExecutor(size, r -> { - Thread thread = ScheduledExecutorScheduler.this.thread = new Thread(threadGroup, r, name); + Thread thread = ScheduledExecutorScheduler.this.thread = new Thread(threadGroup, r, name + "-" + count.incrementAndGet()); thread.setDaemon(daemon); thread.setContextClassLoader(classloader); return thread; From c37c62323c311afd7bbd07d15927b2fd5a65d5ec Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Tue, 17 Sep 2019 12:41:06 +1000 Subject: [PATCH 2/2] updates from review Signed-off-by: Greg Wilkins --- .../jetty/util/thread/ScheduledExecutorScheduler.java | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java b/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java index 90a629b5205..891b6f6d9f7 100644 --- a/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java +++ b/jetty-util/src/main/java/org/eclipse/jetty/util/thread/ScheduledExecutorScheduler.java @@ -24,7 +24,7 @@ import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicInteger; -import org.eclipse.jetty.util.ProcessorUtils; +import org.eclipse.jetty.util.StringUtil; import org.eclipse.jetty.util.annotation.Name; import org.eclipse.jetty.util.component.AbstractLifeCycle; import org.eclipse.jetty.util.component.Dumpable; @@ -83,7 +83,7 @@ public class ScheduledExecutorScheduler extends AbstractLifeCycle implements Sch */ public ScheduledExecutorScheduler(@Name("name") String name, @Name("daemon") boolean daemon, @Name("classLoader") ClassLoader classLoader, @Name("threadGroup") ThreadGroup threadGroup, @Name("threads") int threads) { - this.name = name == null ? "Scheduler-" + hashCode() : name; + this.name = StringUtil.isBlank(name) ? "Scheduler-" + hashCode() : name; this.daemon = daemon; this.classloader = classLoader == null ? Thread.currentThread().getContextClassLoader() : classLoader; this.threadGroup = threadGroup; @@ -93,7 +93,7 @@ public class ScheduledExecutorScheduler extends AbstractLifeCycle implements Sch @Override protected void doStart() throws Exception { - int size = threads > 0 ? threads : Math.max(1, ProcessorUtils.availableProcessors() / 4); + int size = threads > 0 ? threads : 1; scheduler = new ScheduledThreadPoolExecutor(size, r -> { Thread thread = ScheduledExecutorScheduler.this.thread = new Thread(threadGroup, r, name + "-" + count.incrementAndGet());