TestNG vs JUnit 5 battle

Sergey Pirogov

Senior Automation QA

@s_pirogov

http://automation-remarks.com

Java is on top

Java - it is reliable

49%

Python - easy and fast

21%

Same as main project

17%

Other

10%

http://automated-testing.info/t/opros-na-kakom-yazyke-delat-avtomatizacziyu/12252

People choose TestNG

65%

35%

TestNG

JUnit

http://automated-testing.info/t/junit-vs-testng-nuzhna-pomoshh-ot-kommyuniti/11988

JUnit 4 kung fu

@Test
public void pushTest() {
   System.out.println("Parameterized Number is : " + number);
}
@RunWith(value = Parameterized.class)
public class JunitTest6 {

	 private int number;

	 public JunitTest(int number) {
	    this.number = number;
	 }

	 @Parameters
	 public static Collection<Object[]> data() {
	   Object[][] data = new Object[][] { { 1 }, { 2 } };
	   return Arrays.asList(data);
	 }
        
         @Rule
         public ExpectedException expectedEx = ExpectedException.none();
         @Rule         
         public VideoRule video = new VideoRule();

         @Test
         @Video
         public void pushTest() {
            expectedEx.expect(RuntimeException.class);
            expectedEx.expectMessage("Employee ID is null");
            System.out.println("Parameterized Number is : " + number);
         }
}
  • Parametrize test
  • Check exception type and message
  • Add Video Recorder Support
@Listeners(VideoListener.class)
public class TestNGParametrizedTest {

  @DataProvider
  public static Object[][] number() {
    return new Object[][]{{1}, {2}, {3}};
  }

  @Video
  @Test(expectedExceptions = RuntimeException.class,
          expectedExceptionsMessageRegExp = "Employee ID is null",
          dataProvider = "number")
  public void thisIsParametrized(int number) {
    System.out.println("Parameterized Number is : " + number);
  }
}

TestNG kung fu

TestNG is widely used because...

  • Beloved by a lot of AQA  
  • Easier test parametrization
  • Parallel test run support    
  • Have a lot of point to extend
  • A lot of training materials     

Is there any chance Junit 5 beat TestNG?

Public is not required!

class JUnit5Test {

    @Test
    void someTest() {
        assertTrue(true);
    }
}

New annotation names

@BeforeClass
@Before
@AfterClass
@After

Before

Now

@BeforeAll
@BeforeEach
@AfterEach
@AfterAll

Composed Annotations

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Tag("Fast")
@Tag("UI")
@Test
public @interface FastUiTest {}
@Test
@Tag("UI")
@Tag("Fast")
void fastTest(){
    assertTrue(false);
}
@FastUiTest
void fastTest(){
    assertTrue(false);
}

Before

Now

Soft Assertions

@Test
void testWithSoftAssert() {
    User user = new User("Dima", 27);

    assertAll(
            () -> assertEquals("Dma", user.getName()),
            () -> assertEquals(26, user.getAge())
    );
}

org.opentest4j.MultipleFailuresError: Multiple Failures (2 failures)
    expected: <Dma> but was: <Dima>
    expected: <26> but was: <27>

private SoftAssert softAssert = new SoftAssert();

@Test
public void testForSoftAssertionFailure() {
  softAssert.assertTrue(false);
  softAssert.assertEquals(1, 2);
  softAssert.assertAll();
}

Extension  points

Before

Now

  • RunWith
  • Rule
  • ClassRule
  • TestWatcher
  • Test Instance Post Processor
  • BeforeAll Callback
  • Test Execution Condition
  • BeforeEach Callback
  • Parameter Resolution
  • Before Test Execution
  • After Test Execution
  • Exception Handling
  • AfterEach Callback
  • AfterAll Callback

JUnit DisableIfOS

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@ExtendWith(DisableIfCondition.class)
public @interface DisableIfOS {
  String value();
}
public class DisableIfCondition implements TestExecutionCondition {
  @Override
  public ConditionEvaluationResult evaluate(TestExtensionContext context) {
     // logic here
  }
}

Junit DisableIfOs

public class DisabledTest {

  @Test
  @DisableIfOS(WINDOWS)
  void thisIsDisabledOnWindows() {
    assert false;
  }
}

TestNG DisableIfOs

@Listeners(DisableTransformer.class)
public class DisabledTest {

  @Test
  @DisableIfOS(WINDOWS)
  void thisIsDisabledOnWindows() {
    assert false;
  }
}
public class DisableTransformer implements IAnnotationTransformer {
  @Override
  public void transform(ITestAnnotation annotation, 
                        Class testClass, 
                        Constructor testConstructor, 
                        Method testMethod) {
    if (isDisabled(testMethod)) {
      annotation.setEnabled(false);
    }
  }

  private boolean isDisabled(Method testMethod) {
    DisableIfOs annotation = testMethod
                               .getDeclaredAnnotation(DisableIfOs.class);
    return annotation != null && System.getProperty("os.name")
                               .startsWith(annotation.value());
  }
}

Junit 5 Parameter injection

@ExtendsWith(UserParameterResolver.class)
class Parametrized{

    @Test
    void canRegister(User user) {
    // do something with `server`
    }
}
public class UserParameterResolver implements ParameterResolver {
  @Override
  public boolean supports(ParameterContext parameterContext, 
                          ExtensionContext extensionContext){
    return parameterContext.getParameter().getType().equals(User.class);
  }

  @Override
  public Object resolve(ParameterContext parameterContext, 
                        ExtensionContext extensionContext){
    return new User("Ivan");
  }
}

TestNG parameter injection

  @Test(dataProvider = "user", 
        dataProviderClass = UserDataProvider.class)
  public void register(User user) {
    assert user.getName().equals("Ivan");
  }
public class ParameterTransformer implements IAnnotationTransformer {
  @Override
  public void transform(ITestAnnotation annotation, 
                        Class testClass,
                        Constructor testConstructor, 
                        Method testMethod) {

    Class<?>[] parameterTypes = testMethod.getParameterTypes();

    if (Arrays.asList(parameterTypes).contains(User.class)) {
      annotation.setDataProviderClass(UserDataProvider.class);
      annotation.setDataProvider("user");
    }

  }

TestNG smart injection

@Listeners(ParameterTransformer.class)
class Test{  

  @Test
  public void register(User user) {
    assert user.getName().equals("Ivan");
  }

}

Test Execution Listeners

public class VideoExtension implements BeforeTestExecutionCallback, 
                                       AfterTestExecutionCallback {

  private IVideoRecorder recorder;

  @Override
  public void beforeTestExecution(TestExtensionContext context) throws Exception {
    recorder = RecorderFactory.getRecorder(VideoRecorder.conf().getRecorderType());
    recorder.start();
  }

  @Override
  public void afterTestExecution(TestExtensionContext context) throws Exception {
    File video = stopRecording(fileName);
  }

JUnit 5 Apply Listeners

@ExtendWith(VideoExtension.class)
public class AutomaticDriverManagerTest {

     @Test
     void testCanDoMagic(){

     }
}
public class AutomaticDriverManagerTest {

     @Test
     @ExtendWith(VideoExtension.class)
     void testCanDoMagic(){

     }
}
public class AutomaticDriverManagerTest {

     @Test
     @Video
     void testCanDoMagic(){

     }
}

JUnit 5 Extensions

Summary

  • flexible because of many extension points
  • extensions compose well
  • customizable due to meta-annotations

TestNG listeners bug

@Listeners(VideoListener.class)
public class TestNgVideoTest {

    @Test
    @Video
    public void shouldFail() {
        Thread.sleep(1000);
        assert false;
    }
}

https://github.com/cbeust/testng/issues/88

JUnit 5 Build tools support

apply plugin: 'org.junit.platform.gradle.plugin'

junitPlatform {
    // filter tests to run
    selectors {
        packages 'com.acme.foo', 'com.acme.bar'
        methods 'com.acme.Foo#a', 'com.acme.Foo#b'
        method 'com.example.app.Application#run(java.lang.String[])'
    }
    //

    tags {
        include 'fast', 'smoke'
        exclude 'slow', 'ci'
    }

    packages {
        include 'com.sample.included1', 'com.sample.included2'
        exclude 'com.sample.excluded1', 'com.sample.excluded2'
    }   
}

TestNG Build tools support

apply plugin: 'java'

test {
     useTestNG {
         suites 'src/main/resources/testng.xml'
     }
}

Boilerplate is gone

public class JUnit5Parametrized {
  
  @Video
  @ParametrizedTest
  public void pushTest(Integer number) {
    Throwable exception = assertThrows(RuntimeException.class, () -> {
      System.out.println(getUserById(number));
    });
    assertEquals("Employee ID is null",exception.getMessage());
  }
}
@RunWith(value = Parameterized.class)
public class Junit4Parametrized {

	 private int number;

	 public JunitTest(int number) {
	    this.number = number;
	 }

	 @Parameters
	 public static Collection<Object[]> data() {
	   Object[][] data = new Object[][] { { 1 }, { 2 } };
	   return Arrays.asList(data);
	 }
        
         @Rule
         public ExpectedException expectedEx = ExpectedException.none();
         @Rule    
         public VideoRule video = new VideoRule();

         @Video
         @Test
         public void pushTest() {
            expectedEx.expect(RuntimeException.class);
            expectedEx.expectMessage("Employee ID is null");
            System.out.println("Parameterized Number is : " + number);
         }
}

Summary

JUnit5​

Extension

TestNG

Parameters

Parallel testes

Build tools

support

Test tags

Production

JUnit 5 is a great alternative for TestNG !

Thank you!

@s_pirogov

http://automation-remarks.com

TestNG vs JUnit battle

By Sergey Pirogov

TestNG vs JUnit battle

  • 16,002