如何使用 junit5 和 testcontainers 测试存储库?
Posted
技术标签:
【中文标题】如何使用 junit5 和 testcontainers 测试存储库?【英文标题】:How to test repository with junit5 and testcontainers? 【发布时间】:2021-02-10 16:21:15 【问题描述】:我有一个示例项目,我在其中试验不同的技术。
我有以下设置:
Spring Boot 2.3.4.RELEASE 飞路 7.0.1 测试容器 1.15.0-rc2 Junit 5.7.0如何使用 testcontainer-junit5 测试 Repository 层?
我现在为CompanyRepositoryTest.java
提供的代码示例:
@ExtendWith(SpringExtension.class)
@Testcontainers
public class CompanyRepositoryTest
@Autowired
private CompanyRepository companyRepository;
@Container
public mysqlContainer mysqlContainer = new MySQLContainer()
.withDatabaseName("foo")
.withUsername("foo")
.withPassword("secret");;
@Test
public void whenFindByIdExecuted_thenNullReturned()
throws Exception
assertEquals(companyRepository.findById(1L), Optional.ofNullable(null));
@Test
public void whenFindAllExecuted_thenEmptyListReturned()
assertEquals(companyRepository.findAll(), new ArrayList<>());
当我添加@SpringBootTest
时,我需要设置所有上下文并且有一些应用程序加载上下文问题?
问题是,谁能揭开@TestContainers
注释的作用?在测试存储库时使用它的最佳做法或正确做法是什么?
【问题讨论】:
除了已经很好的答案外,本指南还讨论了 Testcontainers + Spring Boot + JUnit (4/5) 的所有可能设置方式:rieckpil.de/… 【参考方案1】:@Testcontainers
注释提供的JUnit 5 extension 会扫描使用@Container
注释声明的任何容器,然后启动和停止这些容器以进行测试。作为静态字段的容器将与所有测试共享,作为实例字段的容器将为每个测试启动和停止。
如果您使用 Spring Boot,为测试设置测试容器的最简单方法可能是在 application-test.yml
中提供属性。这将使用数据源 JDBC URL 来启动 testcontainers 容器。有关详细信息,请参阅 Testcontainers JDBC support。
您也可以使用@DataJpaTest
而不是@SpringBootTest
仅测试存储库层:
@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@ActiveProfiles("test")
class CompanyRepositoryTest
您的application-test.yml
文件:
spring:
datasource:
url: jdbc:tc:mysql:8.0://hostname/databasename
driver-class-name: org.testcontainers.jdbc.ContainerDatabaseDriver
在某些情况下,您可能还想使用@TestPropertySource
注释:
@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@TestPropertySource(
properties =
"spring.datasource.url = jdbc:tc:mysql:8.0://hostname/test-database",
"spring.datasource.driver-class-name = org.testcontainers.jdbc.ContainerDatabaseDriver"
)
class CompanyRepositoryTest
请注意,hostname
和 test-database
实际上并未在任何地方使用。
【讨论】:
【参考方案2】:你说
当我添加@SpringBootTest 时,我需要设置所有上下文并拥有 一些应用程序加载上下文问题?
如果您想尝试替代方案并且 Testcontainer 不是强制性的,您可以采用不同的方式。
使用SpringBootTest注解不需要加载everyting,可以指定需要哪些类如
@SpringBootTest(classes = TheService.class )
或者使用@Import
注解
并嘲笑其他人,例如
@MockBean
MyService service;
对于数据库连接,您可以使用注释,例如
@ActiveProfiles("my-profile-for-jpa-test")
@DataJpaTest
@EnableJpaAuditing
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
编辑:我觉得这应该是一个评论,但我想用正确的格式解决问题的 SpringBootTest 部分
【讨论】:
这是一个不错的答案,但我也想使用测试容器。作为我想要评估的技术之一。另外,我特别想要 testcontainers + junit5,因为我使用过 testconatiners + junit4,并且网上有很多关于它的用法的参考资料,但还不是 junit5。【参考方案3】:这是一个例子,我是如何在 Spring 中使用 MySql 配置 Liquibase(类似于 Flyway 的框架):
@DataJpaTest
@TestPropertySource(properties = "spring.jpa.hibernate.ddl-auto=validate")
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@ContextConfiguration(initializers = MySqlLiquibaseBaseIT.Initializer.class )
@Testcontainers
public class MySqlLiquibaseBaseIT
@Container
public static MySQLContainer<?> mysql = new MySQLContainer<>(
DockerImageName
.parse(MySQLContainer.NAME)
.withTag("5.7.22"));
@Configuration
@EnableJpaRepositories
@EntityScan
static class Initializer implements ApplicationContextInitializer<ConfigurableApplicationContext>
@Override
public void initialize(ConfigurableApplicationContext configurableApplicationContext)
TestPropertyValues.of(
"spring.datasource.url=" + mysql.getJdbcUrl(),
"spring.datasource.username=" + mysql.getUsername(),
"spring.datasource.password=" + mysql.getPassword(),
"spring.datasource.driver-class-name=" + mysql.getDriverClassName())
.applyTo(configurableApplicationContext.getEnvironment());
@Bean
public SpringLiquibase springLiquibase(DataSource dataSource)
SpringLiquibase liquibase = new SpringLiquibase();
liquibase.setDropFirst(true);
liquibase.setDataSource(dataSource);
liquibase.setChangeLog("classpath:/db/changelog/db.changelog-master.yml");
return liquibase;
完整的MySqlLiquibaseBaseIT.java
【讨论】:
【参考方案4】:-
根据docs:
测试容器扩展查找所有带有注释的字段 Container 并调用它们的容器生命周期方法。容器 声明为静态字段将在测试方法之间共享。他们 在执行任何测试方法之前只会启动一次,并且 在最后一个测试方法执行后停止。声明的容器 因为每个测试方法都会启动和停止实例字段。
因此,在您的情况下,它将为每个测试方法重新创建一个容器,它只负责启动和停止容器。如果您需要一些测试数据 - 必须手动完成,我看到您有 Flyway,应该这样做。
你在说什么“上下文问题”?
存储库通常不单独测试,您可以只测试运行存储库方法的服务,而不是为两者编写测试。如果您仍然想测试 repos - 用 @Before
中的一些数据填充数据库。
如果您有更多问题,请提出。
【讨论】:
单独集成测试存储库(或取决于您的架构,持久层)非常好。这就是 @DataJpaTest 的用途。以上是关于如何使用 junit5 和 testcontainers 测试存储库?的主要内容,如果未能解决你的问题,请参考以下文章
如何使用 JUnit 5 在 Kotlin 中创建 TestContainers 基测试类
使用 Micronaut 应用程序为 JUnit5 中的每个单元/集成测试运行一次 TestContainer
如何使用 Junit 5 使用 Testcontainer 对 Kafka 运行集成测试
如何将 Testcontainers 与 @DataJpaTest 结合使用以避免代码重复?