I am using JUnit 5.7.0. I have a class that has a set-up method that I want to run before each nested classes and a tear-down method that I want to run after each nested class, but not before and after each test method within the nested classes.
Currently, the code looks something like this:
class BatchJob0TestClass
{
@BeforeEach
public void setUp()
{
//set up test data
}
@AfterEach
public void tearDown()
{
//tear down test data
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given only good records in the database")
class AllGoodRecordsTest
{
@Test
@Order(1)
@DisplayName("Then the job should finish successfully")
void shouldSucceed()
{
//run batch job and check that it exited as successful
}
@Test
@Order(2)
@DisplayName("Then some value X should change from '0' to '1' for all test records")
void shouldChangeSomeValueXFrom0To1ForAllTestRecords()
{
//verify that some value X changed from '0' to '1' for all test records
}
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given one record in the database where the data is bad")
class OneRecordIsBadTest
{
@BeforeEach
void setup()
{
//change default test set up to have the data of one record be bad
}
@Test
@DisplayName("Then the job should finish as failed")
void shouldFail()
{
//run batch job and verify that it exited as failed
}
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given another record in the database where the data is bad in a different way")
class AnotherRecordIsBadInADifferentWayTest
{
@BeforeEach
void setup()
{
//change default test set up to have another record be bad in a different way
}
@Test
@DisplayName("Then the job should finish as failed")
void shouldFail()
{
//run batch job and verify that it exited as failed
}
}
}
Another iteration of the above used @BeforeAll and @AfterAll, respectively, for the setUp() and tearDown() methods.
class BatchJob0TestClass
{
@BeforeAll
public static void setUp()
{
//set up test data
}
@AfterAll
public static void tearDown()
{
//tear down test data
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given only good records in the database")
class AllGoodRecordsTest
{
@Test
@Order(1)
@DisplayName("Then the job should finish successfully")
void shouldSucceed()
{
//run batch job and check that it exited as successful
}
@Test
@Order(2)
@DisplayName("Then some value X should change from '0' to '1' for all test records")
void shouldChangeSomeValueXFrom0To1ForAllTestRecords()
{
//verify that some value X changed from '0' to '1' for all test records
}
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given one record in the database where the data is bad")
class OneRecordIsBadTest
{
@BeforeEach
void setup()
{
//change default test set up to have the data of one record be bad
}
@Test
@DisplayName("Then the job should finish as failed")
void shouldFail()
{
//run batch job and verify that it exited as failed
}
}
@Nested
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("Given another record in the database where the data is bad in a different way")
class AnotherRecordIsBadInADifferentWayTest
{
@BeforeEach
void setup()
{
//change default test set up to have another record be bad in a different way
}
@Test
@DisplayName("Then the job should finish as failed")
void shouldFail()
{
//run batch job and verify that it exited as failed
}
}
}
Regardless how I do it, I can’t get the desired behavior. Either the test data is set up or torn down once each for the outer class only, or it is set up and torn down for every test method.
Is what I’m trying to achieve even possible using JUnit’s out-of-the-box annotations, or do I just need to resort to calls to unannotated helper methods to get the setUp() and tearDown() methods to run only at the start and end of the test classes and not each test? I know how to do the latter, but I was hoping that JUnit was capable of the desired behavior through annotations alone.
Nonnemo is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.