blob: 96e7170b94a0bade349d618904ce2f13eca7c327 [file] [log] [blame]
/*
* 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.maven.surefire.testng;
import java.io.File;
import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger;
import org.apache.maven.surefire.api.booter.ProviderParameterNames;
import org.apache.maven.surefire.api.cli.CommandLineOption;
import org.apache.maven.surefire.api.report.RunListener;
import org.apache.maven.surefire.api.testset.TestListResolver;
import org.apache.maven.surefire.api.testset.TestSetFailedException;
import org.apache.maven.surefire.shared.utils.StringUtils;
import org.apache.maven.surefire.testng.conf.Configurator;
import org.apache.maven.surefire.testng.utils.FailFastEventsSingleton;
import org.apache.maven.surefire.testng.utils.FailFastListener;
import org.apache.maven.surefire.testng.utils.FailFastNotifier;
import org.apache.maven.surefire.testng.utils.Stoppable;
import org.testng.ITestNGListener;
import org.testng.TestNG;
import org.testng.annotations.Test;
import org.testng.xml.XmlClass;
import org.testng.xml.XmlMethodSelector;
import org.testng.xml.XmlSuite;
import org.testng.xml.XmlTest;
import static org.apache.maven.surefire.api.cli.CommandLineOption.LOGGING_LEVEL_DEBUG;
import static org.apache.maven.surefire.api.cli.CommandLineOption.SHOW_ERRORS;
import static org.apache.maven.surefire.api.util.ReflectionUtils.instantiate;
import static org.apache.maven.surefire.api.util.ReflectionUtils.invokeSetter;
import static org.apache.maven.surefire.api.util.ReflectionUtils.newInstance;
import static org.apache.maven.surefire.api.util.ReflectionUtils.tryGetConstructor;
import static org.apache.maven.surefire.api.util.ReflectionUtils.tryGetMethod;
import static org.apache.maven.surefire.api.util.ReflectionUtils.tryLoadClass;
import static org.apache.maven.surefire.api.util.internal.ConcurrencyUtils.runIfZeroCountDown;
/**
* Contains utility methods for executing TestNG.
*
* @author <a href="mailto:brett@apache.org">Brett Porter</a>
* @author <a href='mailto:the[dot]mindstorm[at]gmail[dot]com'>Alex Popescu</a>
*/
final class TestNGExecutor {
/** The default name for a suite launched from the maven surefire plugin */
private static final String DEFAULT_SUREFIRE_SUITE_NAME = "Surefire suite";
/** The default name for a test launched from the maven surefire plugin */
private static final String DEFAULT_SUREFIRE_TEST_NAME = "Surefire test";
private static final boolean HAS_TEST_ANNOTATION_ON_CLASSPATH =
tryLoadClass(TestNGExecutor.class.getClassLoader(), "org.testng.annotations.Test") != null;
// Using reflection because XmlClass.setIndex is available since TestNG 6.3
// XmlClass.m_index field is available since TestNG 5.13, but prior to 6.3 required invoking constructor
// and constructor XmlClass constructor signatures evolved over time.
private static final Method XML_CLASS_SET_INDEX = tryGetMethod(XmlClass.class, "setIndex", int.class);
// For TestNG versions [5.13, 6.3) where XmlClass.setIndex is not available, invoke XmlClass(String, boolean, int)
// constructor. Note that XmlClass(String, boolean, int) was replaced with XmlClass(String, int) when
// XmlClass.setIndex already existed.
private static final Constructor<XmlClass> XML_CLASS_CONSTRUCTOR_WITH_INDEX =
tryGetConstructor(XmlClass.class, String.class, boolean.class, int.class);
private TestNGExecutor() {
throw new IllegalStateException("not instantiable constructor");
}
@SuppressWarnings("checkstyle:parameternumbercheck")
static void run(
Iterable<Class<?>> testClasses,
String testSourceDirectory,
Map<String, String> options, // string,string because TestNGMapConfigurator#configure()
TestNGReporter testNGReporter,
File reportsDirectory,
TestListResolver methodFilter,
List<CommandLineOption> mainCliOptions,
int skipAfterFailureCount)
throws TestSetFailedException {
TestNG testng = new TestNG(true);
Configurator configurator = getConfigurator(options.get("testng.configurator"));
if (isCliDebugOrShowErrors(mainCliOptions)) {
System.out.println(
"Configuring TestNG with: " + configurator.getClass().getSimpleName());
}
XmlMethodSelector groupMatchingSelector = createGroupMatchingSelector(options);
XmlMethodSelector methodNameFilteringSelector = createMethodNameFilteringSelector(methodFilter);
Map<String, SuiteAndNamedTests> suitesNames = new HashMap<>();
List<XmlSuite> xmlSuites = new ArrayList<>();
for (Class<?> testClass : testClasses) {
TestMetadata metadata = findTestMetadata(testClass);
SuiteAndNamedTests suiteAndNamedTests = suitesNames.get(metadata.suiteName);
if (suiteAndNamedTests == null) {
suiteAndNamedTests = new SuiteAndNamedTests();
suiteAndNamedTests.xmlSuite.setName(metadata.suiteName);
configurator.configure(suiteAndNamedTests.xmlSuite, options);
xmlSuites.add(suiteAndNamedTests.xmlSuite);
suitesNames.put(metadata.suiteName, suiteAndNamedTests);
}
XmlTest xmlTest = suiteAndNamedTests.testNameToTest.get(metadata.testName);
if (xmlTest == null) {
xmlTest = new XmlTest(suiteAndNamedTests.xmlSuite);
xmlTest.setName(metadata.testName);
addSelector(xmlTest, groupMatchingSelector);
addSelector(xmlTest, methodNameFilteringSelector);
xmlTest.setXmlClasses(new ArrayList<>());
suiteAndNamedTests.testNameToTest.put(metadata.testName, xmlTest);
}
xmlTest.getXmlClasses()
.add(newXmlClassInstance(
testClass.getName(), xmlTest.getXmlClasses().size()));
}
testng.setXmlSuites(xmlSuites);
configurator.configure(testng, options);
postConfigure(
testng,
testSourceDirectory,
testNGReporter,
reportsDirectory,
skipAfterFailureCount,
extractVerboseLevel(options));
testng.run();
}
private static XmlClass newXmlClassInstance(String testClassName, int index) {
// In case of parallel test execution with parallel="methods", TestNG orders test execution
// by XmlClass.m_index field. When unset (equal for all XmlClass instances), TestNG can
// invoke `@BeforeClass` setup methods on many instances, without invoking `@AfterClass`
// tearDown methods, thus leading to high resource consumptions when test instances
// allocate resources.
// With index set, order of setup, test and tearDown methods is reasonable, with approximately
// #thread-count many test classes being initialized at given point in time.
// Note: XmlClass.m_index field is set automatically by TestNG when it reads a suite file.
if (XML_CLASS_SET_INDEX != null) {
XmlClass xmlClass = new XmlClass(testClassName);
invokeSetter(xmlClass, XML_CLASS_SET_INDEX, index);
return xmlClass;
}
if (XML_CLASS_CONSTRUCTOR_WITH_INDEX != null) {
boolean loadClass = true; // this is the default
return newInstance(XML_CLASS_CONSTRUCTOR_WITH_INDEX, testClassName, loadClass, index);
}
return new XmlClass(testClassName);
}
private static boolean isCliDebugOrShowErrors(List<CommandLineOption> mainCliOptions) {
return mainCliOptions.contains(LOGGING_LEVEL_DEBUG) || mainCliOptions.contains(SHOW_ERRORS);
}
private static TestMetadata findTestMetadata(Class<?> testClass) {
TestMetadata result = new TestMetadata();
if (HAS_TEST_ANNOTATION_ON_CLASSPATH) {
Test testAnnotation = findAnnotation(testClass, Test.class);
if (null != testAnnotation) {
if (!StringUtils.isBlank(testAnnotation.suiteName())) {
result.suiteName = testAnnotation.suiteName();
}
if (!StringUtils.isBlank(testAnnotation.testName())) {
result.testName = testAnnotation.testName();
}
}
}
return result;
}
private static <T extends Annotation> T findAnnotation(Class<?> clazz, Class<T> annotationType) {
if (clazz == null) {
return null;
}
T result = clazz.getAnnotation(annotationType);
if (result != null) {
return result;
}
return findAnnotation(clazz.getSuperclass(), annotationType);
}
private static class TestMetadata {
private String testName = DEFAULT_SUREFIRE_TEST_NAME;
private String suiteName = DEFAULT_SUREFIRE_SUITE_NAME;
}
private static class SuiteAndNamedTests {
private final XmlSuite xmlSuite = new XmlSuite();
private final Map<String, XmlTest> testNameToTest = new HashMap<>();
}
private static void addSelector(XmlTest xmlTest, XmlMethodSelector selector) {
if (selector != null) {
xmlTest.getMethodSelectors().add(selector);
}
}
@SuppressWarnings("checkstyle:magicnumber")
private static XmlMethodSelector createMethodNameFilteringSelector(TestListResolver methodFilter)
throws TestSetFailedException {
if (methodFilter != null && !methodFilter.isEmpty()) {
// the class is available in the testClassPath
String clazzName = "org.apache.maven.surefire.testng.utils.MethodSelector";
try {
Class<?> clazz = Class.forName(clazzName);
Method method = clazz.getMethod("setTestListResolver", TestListResolver.class);
method.invoke(null, methodFilter);
} catch (Exception e) {
throw new TestSetFailedException(e.getMessage(), e);
}
XmlMethodSelector xms = new XmlMethodSelector();
xms.setName(clazzName);
// looks to need a high value
xms.setPriority(10000);
return xms;
} else {
return null;
}
}
@SuppressWarnings("checkstyle:magicnumber")
private static XmlMethodSelector createGroupMatchingSelector(Map<String, String> options)
throws TestSetFailedException {
final String groups = options.get(ProviderParameterNames.TESTNG_GROUPS_PROP);
final String excludedGroups = options.get(ProviderParameterNames.TESTNG_EXCLUDEDGROUPS_PROP);
if (groups == null && excludedGroups == null) {
return null;
}
// the class is available in the testClassPath
final String clazzName = "org.apache.maven.surefire.testng.utils.GroupMatcherMethodSelector";
try {
Class<?> clazz = Class.forName(clazzName);
// HORRIBLE hack, but TNG doesn't allow us to setup a method selector instance directly.
Method method = clazz.getMethod("setGroups", String.class, String.class);
method.invoke(null, groups, excludedGroups);
} catch (Exception e) {
throw new TestSetFailedException(e.getMessage(), e);
}
XmlMethodSelector xms = new XmlMethodSelector();
xms.setName(clazzName);
// looks to need a high value
xms.setPriority(9999);
return xms;
}
static void run(
List<String> suiteFiles,
String testSourceDirectory,
Map<String, String> options, // string,string because TestNGMapConfigurator#configure()
TestNGReporter testNGReporter,
File reportsDirectory,
int skipAfterFailureCount)
throws TestSetFailedException {
TestNG testng = new TestNG(true);
Configurator configurator = getConfigurator(options.get("testng.configurator"));
configurator.configure(testng, options);
postConfigure(
testng,
testSourceDirectory,
testNGReporter,
reportsDirectory,
skipAfterFailureCount,
extractVerboseLevel(options));
testng.setTestSuites(suiteFiles);
testng.run();
}
private static Configurator getConfigurator(String className) {
try {
return (Configurator) Class.forName(className).newInstance();
} catch (ReflectiveOperationException e) {
throw new RuntimeException(e);
}
}
private static void postConfigure(
TestNG testNG,
String sourcePath,
TestNGReporter testNGReporter,
File reportsDirectory,
int skipAfterFailureCount,
int verboseLevel) {
// 0 (default): turn off all TestNG output
testNG.setVerbose(verboseLevel);
testNG.addListener((ITestNGListener) testNGReporter);
if (skipAfterFailureCount > 0) {
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
testNG.addListener(instantiate(classLoader, FailFastNotifier.class.getName(), Object.class));
testNG.addListener(
new FailFastListener(createStoppable(testNGReporter.getRunListener(), skipAfterFailureCount)));
}
// FIXME: use classifier to decide if we need to pass along the source dir (only for JDK14)
if (sourcePath != null) {
testNG.setSourcePath(sourcePath);
}
testNG.setOutputDirectory(reportsDirectory.getAbsolutePath());
}
private static Stoppable createStoppable(final RunListener reportManager, int skipAfterFailureCount) {
final AtomicInteger currentFaultCount = new AtomicInteger(skipAfterFailureCount);
return () -> {
runIfZeroCountDown(() -> FailFastEventsSingleton.getInstance().setSkipOnNextTest(), currentFaultCount);
reportManager.testExecutionSkippedByUser();
};
}
private static int extractVerboseLevel(Map<String, String> options) throws TestSetFailedException {
try {
String verbose = options.get("surefire.testng.verbose");
return verbose == null ? 0 : Integer.parseInt(verbose);
} catch (NumberFormatException e) {
throw new TestSetFailedException(
"Provider property 'surefire.testng.verbose' should refer to "
+ "number -1 (debug mode), 0, 1 .. 10 (most detailed).",
e);
}
}
}