为什么 Testcontainers 不使用 application.properties / application.yaml?

问题描述

在我的 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());
        }
    
    }

系统输出

测试
测试
测试

...即使不使用 @DynamicPropertySource

解决方法

容器不知道 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;
,

Testconainer 可以使用 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());
  }
}

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