diff --git a/CHANGES.txt b/CHANGES.txt
index f6810ede1..bcd59d2ed 100644
--- a/CHANGES.txt
+++ b/CHANGES.txt
@@ -1,5 +1,6 @@
Current (7.10.0)
-New: GITHUB-2916: Allow users to define ordering for TestNG listeners (Krishnan Mahadevan)
+Fixed: GITHUB-3066: How to dynamically adjust the number of TestNG threads after IExecutorFactory is deprecated? (Krishnan Mahadevan)
+New: GITHUB-2874: Allow users to define ordering for TestNG listeners (Krishnan Mahadevan)
Fixed: GITHUB-3033: Moved ant support under own repository https://github.com/testng-team/testng-ant (Julien Herr)
Fixed: GITHUB-3064: TestResult lost if failure creating RetryAnalyzer (Krishnan Mahadevan)
Fixed: GITHUB-3048: ConcurrentModificationException when injecting values (Krishnan Mahadevan)
diff --git a/testng-core-api/src/main/java/org/testng/IExecutorServiceFactory.java b/testng-core-api/src/main/java/org/testng/IExecutorServiceFactory.java
new file mode 100644
index 000000000..5636e3790
--- /dev/null
+++ b/testng-core-api/src/main/java/org/testng/IExecutorServiceFactory.java
@@ -0,0 +1,35 @@
+package org.testng;
+
+import java.util.concurrent.BlockingQueue;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.ThreadFactory;
+import java.util.concurrent.TimeUnit;
+
+/**
+ * Represents the capability to create a custom {@link ExecutorService} by downstream consumers. The
+ * implementation can be plugged in via the configuration parameter -threadpoolfactoryclass
+ *
+ */
+@FunctionalInterface
+public interface IExecutorServiceFactory {
+
+ /**
+ * @param corePoolSize the number of threads to keep in the pool, even if they are idle, unless
+ * {@code allowCoreThreadTimeOut} is set
+ * @param maximumPoolSize the maximum number of threads to allow in the pool
+ * @param keepAliveTime when the number of threads is greater than the core, this is the maximum
+ * time that excess idle threads will wait for new tasks before terminating.
+ * @param unit the time unit for the {@code keepAliveTime} argument
+ * @param workQueue the queue to use for holding tasks before they are executed. This queue will
+ * hold only the {@code Runnable} tasks submitted by the {@code execute} method.
+ * @param threadFactory the factory to use when the executor creates a new thread *
+ * @return - An implementation of {@link ExecutorService}
+ */
+ ExecutorService create(
+ int corePoolSize,
+ int maximumPoolSize,
+ long keepAliveTime,
+ TimeUnit unit,
+ BlockingQueue workQueue,
+ ThreadFactory threadFactory);
+}
diff --git a/testng-core-api/src/main/java/org/testng/internal/RuntimeBehavior.java b/testng-core-api/src/main/java/org/testng/internal/RuntimeBehavior.java
index dd9872656..74f9e581f 100644
--- a/testng-core-api/src/main/java/org/testng/internal/RuntimeBehavior.java
+++ b/testng-core-api/src/main/java/org/testng/internal/RuntimeBehavior.java
@@ -29,15 +29,6 @@ public static boolean ignoreCallbackInvocationSkips() {
return Boolean.getBoolean(IGNORE_CALLBACK_INVOCATION_SKIPS);
}
- /**
- * @return - true
if TestNG is to be using its custom implementation of {@link
- * java.util.concurrent.ThreadPoolExecutor} for running concurrent tests. Defaults to
- * false
- */
- public static boolean favourCustomThreadPoolExecutor() {
- return Boolean.getBoolean(FAVOR_CUSTOM_THREAD_POOL_EXECUTOR);
- }
-
/**
* @return - A comma separated list of packages that represent special listeners which users will
* expect to be executed after executing the regular listeners. Here special listeners can be
diff --git a/testng-core/src/main/java/org/testng/SuiteRunner.java b/testng-core/src/main/java/org/testng/SuiteRunner.java
index 7900b1663..bb6baffe1 100644
--- a/testng-core/src/main/java/org/testng/SuiteRunner.java
+++ b/testng-core/src/main/java/org/testng/SuiteRunner.java
@@ -439,7 +439,11 @@ private void runInParallelTestMode() {
}
ThreadUtil.execute(
- "tests", tasks, xmlSuite.getThreadCount(), xmlSuite.getTimeOut(XmlTest.DEFAULT_TIMEOUT_MS));
+ configuration,
+ "tests",
+ tasks,
+ xmlSuite.getThreadCount(),
+ xmlSuite.getTimeOut(XmlTest.DEFAULT_TIMEOUT_MS));
}
private class SuiteWorker implements Runnable {
diff --git a/testng-core/src/main/java/org/testng/SuiteTaskExecutor.java b/testng-core/src/main/java/org/testng/SuiteTaskExecutor.java
index 8a3f3838f..194a66ede 100644
--- a/testng-core/src/main/java/org/testng/SuiteTaskExecutor.java
+++ b/testng-core/src/main/java/org/testng/SuiteTaskExecutor.java
@@ -2,16 +2,12 @@
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
-import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import org.testng.internal.IConfiguration;
-import org.testng.internal.RuntimeBehavior;
import org.testng.internal.Utils;
import org.testng.internal.thread.TestNGThreadFactory;
import org.testng.internal.thread.graph.GraphOrchestrator;
import org.testng.log4testng.Logger;
-import org.testng.thread.IExecutorFactory;
-import org.testng.thread.ITestNGThreadPoolExecutor;
import org.testng.thread.IThreadWorkerFactory;
class SuiteTaskExecutor {
@@ -41,33 +37,18 @@ public SuiteTaskExecutor(
public void execute() {
String name = "suites-";
- if (RuntimeBehavior.favourCustomThreadPoolExecutor()) {
- IExecutorFactory execFactory = configuration.getExecutorFactory();
- ITestNGThreadPoolExecutor executor =
- execFactory.newSuiteExecutor(
- name,
- graph,
- factory,
- threadPoolSize,
- threadPoolSize,
- Integer.MAX_VALUE,
- TimeUnit.MILLISECONDS,
- queue,
- null);
- executor.run();
- service = executor;
- } else {
- service =
- new ThreadPoolExecutor(
- threadPoolSize,
- threadPoolSize,
- Integer.MAX_VALUE,
- TimeUnit.MILLISECONDS,
- queue,
- new TestNGThreadFactory(name));
- GraphOrchestrator executor = new GraphOrchestrator<>(service, factory, graph, null);
- executor.run();
- }
+ service =
+ this.configuration
+ .getExecutorServiceFactory()
+ .create(
+ threadPoolSize,
+ threadPoolSize,
+ Integer.MAX_VALUE,
+ TimeUnit.MILLISECONDS,
+ queue,
+ new TestNGThreadFactory(name));
+ GraphOrchestrator executor = new GraphOrchestrator<>(service, factory, graph, null);
+ executor.run();
}
public void awaitCompletion() {
diff --git a/testng-core/src/main/java/org/testng/TestNG.java b/testng-core/src/main/java/org/testng/TestNG.java
index e02f7ef49..cf188cc26 100644
--- a/testng-core/src/main/java/org/testng/TestNG.java
+++ b/testng-core/src/main/java/org/testng/TestNG.java
@@ -16,6 +16,7 @@
import java.util.Collections;
import java.util.List;
import java.util.Map;
+import java.util.Objects;
import java.util.Optional;
import java.util.ServiceLoader;
import java.util.Set;
@@ -58,7 +59,6 @@
import org.testng.reporters.VerboseReporter;
import org.testng.reporters.XMLReporter;
import org.testng.reporters.jq.Main;
-import org.testng.thread.IExecutorFactory;
import org.testng.thread.IThreadWorkerFactory;
import org.testng.util.Strings;
import org.testng.xml.IPostProcessor;
@@ -151,8 +151,6 @@ public class TestNG {
private final Map, IDataProviderInterceptor>
m_dataProviderInterceptors = Maps.newLinkedHashMap();
- private IExecutorFactory m_executorFactory = null;
-
public static final Integer DEFAULT_VERBOSE = 1;
// Command line suite parameters
@@ -843,10 +841,9 @@ public void setVerbose(int verbose) {
m_verbose = verbose;
}
- /** This method stands deprecated as of TestNG v7.9.0
. */
- @Deprecated
- public void setExecutorFactoryClass(String clazzName) {
- this.m_executorFactory = createExecutorFactoryInstanceUsing(clazzName);
+ public void setExecutorServiceFactory(IExecutorServiceFactory factory) {
+ m_configuration.setExecutorServiceFactory(
+ Objects.requireNonNull(factory, "ExecutorServiceFactory cannot be null"));
}
public void setListenerFactory(ITestNGListenerFactory factory) {
@@ -857,31 +854,6 @@ public void setGenerateResultsPerSuite(boolean generateResultsPerSuite) {
this.m_generateResultsPerSuite = generateResultsPerSuite;
}
- private IExecutorFactory createExecutorFactoryInstanceUsing(String clazzName) {
- Class> cls = ClassHelper.forName(clazzName);
- Object instance = m_objectFactory.newInstance(cls);
- if (instance instanceof IExecutorFactory) {
- return (IExecutorFactory) instance;
- }
- throw new IllegalArgumentException(
- clazzName + " does not implement " + IExecutorFactory.class.getName());
- }
-
- /** This method stands deprecated as of TestNG v7.9.0
. */
- @Deprecated
- public void setExecutorFactory(IExecutorFactory factory) {
- this.m_executorFactory = factory;
- }
-
- /** This method stands deprecated as of TestNG v7.9.0
. */
- @Deprecated
- public IExecutorFactory getExecutorFactory() {
- if (this.m_executorFactory == null) {
- this.m_executorFactory = createExecutorFactoryInstanceUsing(DEFAULT_THREADPOOL_FACTORY);
- }
- return this.m_executorFactory;
- }
-
private void initializeCommandLineSuites() {
if (m_commandLineTestClasses != null || m_commandLineMethods != null) {
if (null != m_commandLineMethods) {
@@ -1018,7 +990,6 @@ private void initializeConfiguration() {
m_configuration.setConfigurable(m_configurable);
m_configuration.setObjectFactory(factory);
m_configuration.setAlwaysRunListeners(this.m_alwaysRun);
- m_configuration.setExecutorFactory(getExecutorFactory());
}
private void addListeners(XmlSuite s) {
@@ -1217,11 +1188,9 @@ public List runSuitesLocally() {
// Create a map with XmlSuite as key and corresponding SuiteRunner as value
for (XmlSuite xmlSuite : m_suites) {
if (m_configuration.isShareThreadPoolForDataProviders()) {
- abortIfUsingGraphThreadPoolExecutor("Shared thread-pool for data providers");
xmlSuite.setShareThreadPoolForDataProviders(true);
}
if (m_configuration.useGlobalThreadPool()) {
- abortIfUsingGraphThreadPoolExecutor("Global thread-pool");
xmlSuite.shouldUseGlobalThreadPool(true);
}
createSuiteRunners(suiteRunnerMap, xmlSuite);
@@ -1272,13 +1241,6 @@ private static void error(String s) {
LOGGER.error(s);
}
- private static void abortIfUsingGraphThreadPoolExecutor(String prefix) {
- if (RuntimeBehavior.favourCustomThreadPoolExecutor()) {
- throw new UnsupportedOperationException(
- prefix + " is NOT COMPATIBLE with TestNG's custom thread pool executor");
- }
- }
-
/**
* @return the verbose level, checking in order: the verbose level on the suite, the verbose level
* on the TestNG object, or 1.
@@ -1514,9 +1476,13 @@ protected void configure(CommandLineArgs cla) {
m_objectFactory.newInstance((Class) clazz));
}
}
- if (cla.threadPoolFactoryClass != null) {
- setExecutorFactoryClass(cla.threadPoolFactoryClass);
- }
+ Optional.ofNullable(cla.threadPoolFactoryClass)
+ .map(ClassHelper::forName)
+ .filter(IExecutorServiceFactory.class::isAssignableFrom)
+ .map(it -> m_objectFactory.newInstance(it))
+ .map(it -> (IExecutorServiceFactory) it)
+ .ifPresent(this::setExecutorServiceFactory);
+
setOutputDirectory(cla.outputDirectory);
String testClasses = cla.testClass;
diff --git a/testng-core/src/main/java/org/testng/TestTaskExecutor.java b/testng-core/src/main/java/org/testng/TestTaskExecutor.java
index 110d8fe92..4818f0eec 100644
--- a/testng-core/src/main/java/org/testng/TestTaskExecutor.java
+++ b/testng-core/src/main/java/org/testng/TestTaskExecutor.java
@@ -3,18 +3,14 @@
import java.util.Comparator;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
-import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.function.Supplier;
import org.testng.internal.IConfiguration;
import org.testng.internal.ObjectBag;
-import org.testng.internal.RuntimeBehavior;
import org.testng.internal.Utils;
import org.testng.internal.thread.TestNGThreadFactory;
import org.testng.internal.thread.graph.GraphOrchestrator;
import org.testng.log4testng.Logger;
-import org.testng.thread.IExecutorFactory;
-import org.testng.thread.ITestNGThreadPoolExecutor;
import org.testng.thread.IThreadWorkerFactory;
import org.testng.xml.XmlTest;
@@ -50,42 +46,27 @@ public TestTaskExecutor(
public void execute() {
String name = "test-" + xmlTest.getName();
int threadCount = Math.max(xmlTest.getThreadCount(), 1);
- if (RuntimeBehavior.favourCustomThreadPoolExecutor()) {
- IExecutorFactory execFactory = configuration.getExecutorFactory();
- ITestNGThreadPoolExecutor executor =
- execFactory.newTestMethodExecutor(
- name,
- graph,
- factory,
- threadCount,
- threadCount,
- 0,
- TimeUnit.MILLISECONDS,
- queue,
- comparator);
- executor.run();
- service = executor;
+ boolean reUse = xmlTest.getSuite().useGlobalThreadPool();
+ Supplier