为啥 Testcontainers 不使用 application.properties / application.yaml?

Posted

技术标签:

【中文标题】为啥 Testcontainers 不使用 application.properties / application.yaml?【英文标题】:Why does Testcontainers not use application.properties / application.yaml?为什么 Testcontainers 不使用 application.properties / application.yaml? 【发布时间】:2021-06-14 13:01:05 【问题描述】:

在我的 Spring Boot 2.4.3 应用程序中,我使用了 Testcontainers,并按照互联网上的说明进行操作。我有一个application.yaml

spring:
  datasource:
    url: jdbc:tc:postgresql:13.2:///testdb?TC_INITSCRIPT=tc_initscript_postgresql.sql
    username: duke
    password: s3crEt
  jpa:
    database-platform: org.hibernate.dialect.PostgreSQL95Dialect
    hibernate:
      ddl-auto: create-drop

但是当我调试应用程序时,容器总是“测试”作为 URL、用户名和密码的值。

这是我的测试类:

@ActiveProfiles("test")
@Testcontainers
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public abstract class AbstractApplicationIT 

    final static DockerImageName POSTGRES_IMAGE = DockerImageName.parse("postgres:13.2-alpine");

        @Container
    //    public static GenericContainer postgreSQLContainer = new GenericContainer(POSTGRES_IMAGE)
        public static PostgreSQLContainer<?> postgreSQLContainer = new PostgreSQLContainer<>(POSTGRES_IMAGE)
    //            .withInitScript("tc_initscript_postgresql.sql")
    //            .withPassword("password")
    //            .withUsername("username")
    //            .withDatabaseName("test")
    //            .withInitScript("tc_initscript_postgresql.sql")
                ;
    
    //    @DynamicPropertySource
    //    static void postgresqlProperties(DynamicPropertyRegistry registry) 
    //        registry.add("spring.datasource.url", postgreSQLContainer::getJdbcUrl);
    //        registry.add("spring.datasource.password", postgreSQLContainer::getPassword);
    //        registry.add("spring.datasource.username", postgreSQLContainer::getUsername);
    //    
    
        @Test
        public void contextLoads() 
            System.out.println(postgreSQLContainer.getDatabaseName());
            System.out.println(postgreSQLContainer.getUsername());
            System.out.println(postgreSQLContainer.getPassword());
        
    
    

System.out:

测试 测试 测试

...即使不使用@DynamicPropertySource

【问题讨论】:

我猜容器是在处理application.properties文件之前启动的。 【参考方案1】:

容器不知道 application.properties。

application.properties 是 Spring Boot 配置,与 Testcontainers 无关。

你可以像这样覆盖值

@Container
private PostgreSQLContainer postgresqlContainer = new PostgreSQLContainer()
    .withDatabaseName("foo")
    .withUsername("foo")
    .withPassword("secret");

您还可以像这样从 application.properties 中读取值:

@Value("spring.datasource.username")
private String username;

【讨论】:

好吧,但我想要的不是硬编码这些值,而是从 application.yaml 中读取它们。 您始终可以使用@Value 读取此值。查看我编辑的答案 好吧,好吧。这意味着我不能在抽象超(测试)类中以静态方式实例化容器,而是必须为每个具体的测试用例创建一个容器。 @DynamicPropertySource 在我看来是一个完全没有意义的注释(至少对于 Testcontainers 或......这种情况)。使用它我总是必须对值进行硬编码。【参考方案2】:

Testconainers 可以使用 spring.datasource.url 的 JDBC support 从您的 spring.datasource.url 派生容器定义。

一个最小的例子如下所示:

@SpringBootTest(properties = 
  "spring.datasource.url=jdbc:tc:postgresql:12:///springboot?TC_INITSCRIPT=somepath/init_mysql.sql"
)
class AbstractApplicationIT 
 
  @Autowired
  private YourRepository yourRepository;
 
  @Test
  @Sql("/scripts/INIT_THREE_ORDERS.sql")
  void shouldReturnOrdersThatContainMacBookPro() 
    List<Order> orders = yourRepository.findAllContainingMacBookPro();
    assertEquals(2, orders.size());
  

为此,请勿在测试中手动指定任何其他容器定义。

【讨论】:

以上是关于为啥 Testcontainers 不使用 application.properties / application.yaml?的主要内容,如果未能解决你的问题,请参考以下文章

如何关闭关闭 TestContainers 中的容器?

如果不存在,Testcontainers 可以为我创建 docker 网络吗?

Testcontainers, Docker in Docker with custom network, 容器不属于网络

使用 testcontainers 测试 kafka 和 spark

Spring testcontainers Driver org.testcontainers.jdbc.ContainerDatabaseDriver 声称不接受 jdbcUrl

尽管存在`@Transactional`,为啥这些数据库修改不回滚?