Use when working with TestNG annotations, assertions, test lifecycle, and configuration for Java testing.
Limited to specific tools
Additional assets for this skill
This skill is limited to using the following tools:
Master TestNG fundamentals including annotations, assertions, test lifecycle, and XML configuration for Java testing. This skill provides comprehensive coverage of essential concepts, patterns, and best practices for professional TestNG development.
TestNG is a powerful testing framework for Java inspired by JUnit and NUnit, designed to cover a wider range of test categories: unit, functional, end-to-end, and integration testing. It supports annotations, data-driven testing, parameterization, and parallel execution.
Add TestNG to your Maven project:
<dependency>
<groupId>org.testng</groupId>
<artifactId>testng</artifactId>
<version>7.9.0</version>
<scope>test</scope>
</dependency>
Configure the Surefire plugin for TestNG:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<version>3.2.5</version>
<configuration>
<suiteXmlFiles>
<suiteXmlFile>testng.xml</suiteXmlFile>
</suiteXmlFiles>
</configuration>
</plugin>
Add TestNG to your Gradle project:
dependencies {
testImplementation 'org.testng:testng:7.9.0'
}
test {
useTestNG()
}
TestNG provides comprehensive lifecycle annotations:
import org.testng.annotations.*;
public class LifecycleTest {
@BeforeSuite
public void beforeSuite() {
// Runs once before the entire test suite
System.out.println("Before Suite");
}
@AfterSuite
public void afterSuite() {
// Runs once after the entire test suite
System.out.println("After Suite");
}
@BeforeTest
public void beforeTest() {
// Runs before each <test> tag in testng.xml
System.out.println("Before Test");
}
@AfterTest
public void afterTest() {
// Runs after each <test> tag in testng.xml
System.out.println("After Test");
}
@BeforeClass
public void beforeClass() {
// Runs once before the first test method in the class
System.out.println("Before Class");
}
@AfterClass
public void afterClass() {
// Runs once after the last test method in the class
System.out.println("After Class");
}
@BeforeMethod
public void beforeMethod() {
// Runs before each test method
System.out.println("Before Method");
}
@AfterMethod
public void afterMethod() {
// Runs after each test method
System.out.println("After Method");
}
@Test
public void testMethod() {
System.out.println("Test Method");
}
}
The @Test annotation supports various attributes:
public class TestAttributesExample {
@Test(description = "Verifies user login functionality")
public void testLogin() {
// Test with description
}
@Test(enabled = false)
public void disabledTest() {
// This test will not run
}
@Test(priority = 1)
public void firstTest() {
// Runs first (lower priority = earlier execution)
}
@Test(priority = 2)
public void secondTest() {
// Runs second
}
@Test(groups = {"smoke", "regression"})
public void groupedTest() {
// Test belongs to multiple groups
}
@Test(dependsOnMethods = {"testLogin"})
public void testDashboard() {
// Runs only if testLogin passes
}
@Test(dependsOnGroups = {"setup"})
public void dependentTest() {
// Runs only if all tests in "setup" group pass
}
@Test(timeOut = 5000)
public void timedTest() {
// Fails if takes more than 5 seconds
}
@Test(invocationCount = 3)
public void repeatedTest() {
// Runs 3 times
}
@Test(invocationCount = 100, threadPoolSize = 10)
public void parallelRepeatedTest() {
// Runs 100 times across 10 threads
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void exceptionTest() {
throw new IllegalArgumentException("Expected");
}
@Test(expectedExceptions = RuntimeException.class,
expectedExceptionsMessageRegExp = ".*invalid.*")
public void exceptionWithMessageTest() {
throw new RuntimeException("This is invalid input");
}
}
TestNG provides comprehensive assertion methods:
import org.testng.Assert;
import org.testng.annotations.Test;
public class AssertionExamples {
@Test
public void testBasicAssertions() {
// Equality
Assert.assertEquals(5, 5);
Assert.assertEquals("hello", "hello");
Assert.assertEquals(new int[]{1, 2, 3}, new int[]{1, 2, 3});
// Boolean
Assert.assertTrue(true);
Assert.assertFalse(false);
// Null checks
Assert.assertNull(null);
Assert.assertNotNull("value");
// Same reference
String s1 = "test";
String s2 = s1;
Assert.assertSame(s1, s2);
Assert.assertNotSame(new String("test"), new String("test"));
}
@Test
public void testAssertionsWithMessages() {
// Assertions with custom failure messages
Assert.assertEquals(5, 5, "Values should be equal");
Assert.assertTrue(true, "Condition should be true");
Assert.assertNotNull("value", "Value should not be null");
}
@Test
public void testCollectionAssertions() {
// Array assertions
String[] expected = {"a", "b", "c"};
String[] actual = {"a", "b", "c"};
Assert.assertEquals(actual, expected);
// Unordered comparison
String[] array1 = {"a", "b", "c"};
String[] array2 = {"c", "a", "b"};
Assert.assertEqualsNoOrder(array1, array2);
}
}
Soft assertions allow multiple assertions to be collected before failing:
import org.testng.annotations.Test;
import org.testng.asserts.SoftAssert;
public class SoftAssertExample {
@Test
public void testWithSoftAssert() {
SoftAssert softAssert = new SoftAssert();
// All assertions are executed
softAssert.assertEquals(1, 1, "First check");
softAssert.assertEquals(2, 3, "Second check - will fail");
softAssert.assertTrue(false, "Third check - will fail");
softAssert.assertNotNull(null, "Fourth check - will fail");
// Report all failures at the end
softAssert.assertAll();
}
}
<!DOCTYPE suite SYSTEM "https://testng.org/testng-1.0.dtd">
<suite name="My Test Suite" verbose="1">
<test name="Unit Tests">
<classes>
<class name="com.example.tests.UserServiceTest"/>
<class name="com.example.tests.ProductServiceTest"/>
</classes>
</test>
<test name="Integration Tests">
<packages>
<package name="com.example.integration.*"/>
</packages>
</test>
</suite>
<!DOCTYPE suite SYSTEM "https://testng.org/testng-1.0.dtd">
<suite name="Group Suite">
<test name="Smoke Tests">
<groups>
<run>
<include name="smoke"/>
</run>
</groups>
<packages>
<package name="com.example.tests.*"/>
</packages>
</test>
<test name="Regression Tests">
<groups>
<run>
<include name="regression"/>
<exclude name="broken"/>
</run>
</groups>
<packages>
<package name="com.example.tests.*"/>
</packages>
</test>
</suite>
<!DOCTYPE suite SYSTEM "https://testng.org/testng-1.0.dtd">
<suite name="Parameterized Suite">
<parameter name="browser" value="chrome"/>
<parameter name="environment" value="staging"/>
<test name="Chrome Tests">
<parameter name="browser" value="chrome"/>
<classes>
<class name="com.example.tests.BrowserTest"/>
</classes>
</test>
<test name="Firefox Tests">
<parameter name="browser" value="firefox"/>
<classes>
<class name="com.example.tests.BrowserTest"/>
</classes>
</test>
</suite>
Using parameters in tests:
import org.testng.annotations.*;
public class BrowserTest {
@Parameters({"browser", "environment"})
@BeforeClass
public void setUp(String browser, @Optional("production") String env) {
System.out.println("Browser: " + browser);
System.out.println("Environment: " + env);
}
@Test
public void testBrowser() {
// Test implementation
}
}
public class GroupExample {
@BeforeGroups("database")
public void setUpDatabase() {
System.out.println("Setting up database");
}
@AfterGroups("database")
public void tearDownDatabase() {
System.out.println("Tearing down database");
}
@Test(groups = {"smoke", "frontend"})
public void testHomePage() {
System.out.println("Testing home page");
}
@Test(groups = {"smoke", "api"})
public void testHealthEndpoint() {
System.out.println("Testing health endpoint");
}
@Test(groups = {"regression", "database"})
public void testDataPersistence() {
System.out.println("Testing data persistence");
}
@Test(groups = {"slow", "integration"})
public void testEndToEnd() {
System.out.println("Testing end-to-end flow");
}
}
public class GroupDependencyExample {
@Test(groups = {"init"})
public void initializeSystem() {
System.out.println("Initializing");
}
@Test(groups = {"init"})
public void configureSystem() {
System.out.println("Configuring");
}
@Test(dependsOnGroups = {"init"}, groups = {"core"})
public void coreTest1() {
System.out.println("Core test 1");
}
@Test(dependsOnGroups = {"init"}, groups = {"core"})
public void coreTest2() {
System.out.println("Core test 2");
}
@Test(dependsOnGroups = {"core"}, groups = {"final"})
public void finalTest() {
System.out.println("Final test");
}
}
import org.testng.*;
public class TestListener implements ITestListener {
@Override
public void onTestStart(ITestResult result) {
System.out.println("Starting: " + result.getName());
}
@Override
public void onTestSuccess(ITestResult result) {
System.out.println("Passed: " + result.getName());
}
@Override
public void onTestFailure(ITestResult result) {
System.out.println("Failed: " + result.getName());
System.out.println("Reason: " + result.getThrowable().getMessage());
}
@Override
public void onTestSkipped(ITestResult result) {
System.out.println("Skipped: " + result.getName());
}
@Override
public void onStart(ITestContext context) {
System.out.println("Test suite starting: " + context.getName());
}
@Override
public void onFinish(ITestContext context) {
System.out.println("Test suite finished: " + context.getName());
}
}
// Using annotation
@Listeners(TestListener.class)
public class MyTest {
@Test
public void testMethod() {
// Test implementation
}
}
Or in testng.xml:
<suite name="Suite">
<listeners>
<listener class-name="com.example.listeners.TestListener"/>
</listeners>
<test name="Test">
<classes>
<class name="com.example.tests.MyTest"/>
</classes>
</test>
</suite>