Большинство аннотаций, связанных с тестированием, можно использовать в качестве мета-аннотаций для создания собственных аннотаций и сокращения конфигурационного дублирования в тестовом комплекте.
Можно использовать каждую из следующих аннотаций в качестве мета-аннотации в сочетании с фреймворком TestContext.
-
@BootstrapWith
-
@ContextConfiguration
-
@ContextHierarchy
-
@ActiveProfiles
-
@TestPropertySource
-
@DirtiesContext
-
@WebAppConfiguration
-
@TestExecutionListeners
-
@Transactional
-
@BeforeTransaction
-
@AfterTransaction
-
@Commit
-
@Rollback
-
@Sql
-
@SqlConfig
-
@SqlMergeMode
-
@SqlGroup
-
@Repeat
(поддерживается только на JUnit 4) -
@Timed
(поддерживается только на JUnit 4) -
@IfProfileValue
(поддерживается только на JUnit 4) -
@ProfileValueSourceConfiguration
(поддерживается только на JUnit 4) -
@SpringJUnitConfig
(поддерживается только на JUnit Jupiter) -
@SpringJUnitWebConfig
(поддерживается только на JUnit Jupiter) -
@TestConstructor
(поддерживается только на JUnit Jupiter) -
@NestedTestConfiguration
(поддерживается только на JUnit Jupiter) -
@EnabledIf
(поддерживается только на JUnit Jupiter) -
@DisabledIf
(поддерживается только на JUnit Jupiter)
Рассмотрим следующий пример:
@RunWith(SpringRunner.class)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
public class OrderRepositoryTests { }
@RunWith(SpringRunner.class)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
public class UserRepositoryTests { }
@RunWith(SpringRunner::class)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
class OrderRepositoryTests { }
@RunWith(SpringRunner::class)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
class UserRepositoryTests { }
Если обнаружится, что предыдущая конфигурация повторяется в нашем тестовом комплекте на базе JUnit 4, то можно сократить дублирование, введя специальную составную аннотацию, которая централизует общую тестовую конфигурацию для Spring, следующим образом:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
public @interface TransactionalDevTestConfig { }
@Target(AnnotationTarget.TYPE)
@Retention(AnnotationRetention.RUNTIME)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
annotation class TransactionalDevTestConfig { }
Затем можно использовать нашу специальную аннотацию @TransactionalDevTestConfig
для упрощения конфигурации отдельных тестовых классов на основе JUnit 4, как показано ниже:
@RunWith(SpringRunner.class)
@TransactionalDevTestConfig
public class OrderRepositoryTests { }
@RunWith(SpringRunner.class)
@TransactionalDevTestConfig
public class UserRepositoryTests { }
@RunWith(SpringRunner::class)
@TransactionalDevTestConfig
class OrderRepositoryTests
@RunWith(SpringRunner::class)
@TransactionalDevTestConfig
class UserRepositoryTests
Если мы пишем тесты, использующие JUnit Jupiter, то можно еще больше сократить дублирование кода, поскольку аннотации в JUnit 5 могут также использоваться в качестве мета-аннотаций. Рассмотрим следующий пример:
@ExtendWith(SpringExtension.class)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
class OrderRepositoryTests { }
@ExtendWith(SpringExtension.class)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
class UserRepositoryTests { }
@ExtendWith(SpringExtension::class)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
class OrderRepositoryTests { }
@ExtendWith(SpringExtension::class)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
class UserRepositoryTests { }
Если обнаружится, что предыдущая конфигурация повторяется в нашем тестовом комплекте на базе JUnit Jupiter, то можно сократить дублирование, введя специальную составную аннотацию, которая централизует общую тестовую конфигурацию для Spring и JUnit Jupiter, следующим образом:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@ExtendWith(SpringExtension.class)
@ContextConfiguration({"/app-config.xml", "/test-data-access-config.xml"})
@ActiveProfiles("dev")
@Transactional
public @interface TransactionalDevTestConfig { }
@Target(AnnotationTarget.TYPE)
@Retention(AnnotationRetention.RUNTIME)
@ExtendWith(SpringExtension::class)
@ContextConfiguration("/app-config.xml", "/test-data-access-config.xml")
@ActiveProfiles("dev")
@Transactional
annotation class TransactionalDevTestConfig { }
Затем можно использовать нашу специальную аннотацию @TransactionalDevTestConfig
для упрощения конфигурации отдельных тестовых классов на основе JUnit Jupite, как показано ниже:
@TransactionalDevTestConfig
class OrderRepositoryTests { }
@TransactionalDevTestConfig
class UserRepositoryTests { }
@TransactionalDevTestConfig
class OrderRepositoryTests { }
@TransactionalDevTestConfig
class UserRepositoryTests { }
Поскольку JUnit Jupiter поддерживает использование @Test
, @RepeatedTest
, ParameterizedTest
и других в качестве мета-аннотаций, можно также создавать собственные составные аннотации на уровне тестового метода. Например, если нужно создать составную аннотацию, которая объединяет аннотации @Test
и @Tag
из JUnit Jupiter с аннотацией @Transactional
из Spring, можно создать аннотацию @TransactionalIntegrationTest
следующим образом:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Transactional
@Tag("integration-test") // org.junit.jupiter.api.Tag
@Test // org.junit.jupiter.api.Test
public @interface TransactionalIntegrationTest { }
@Target(AnnotationTarget.TYPE)
@Retention(AnnotationRetention.RUNTIME)
@Transactional
@Tag("integration-test") // org.junit.jupiter.api.Tag
@Test // org.junit.jupiter.api.Test
annotation class TransactionalIntegrationTest { }
Затем можно использовать нашу специальную аннотацию @TransactionalIntegrationTest
для упрощения конфигурации отдельных тестовых методов на основе JUnit Jupiter следующим образом:
@TransactionalIntegrationTest
void saveOrder() { }
@TransactionalIntegrationTest
void deleteOrder() { }
@TransactionalIntegrationTest
fun saveOrder() { }
@TransactionalIntegrationTest
fun deleteOrder() { }
Более подробную информацию можно найти на вики-странице Spring Annotation Programming Model.
ПЕРЕЙДИТЕ В ПОЛНУЮ ВЕРСИЮ