问题描述
在我的 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());
}
}
为此,请勿在测试中手动指定任何其他容器定义。