首页 > 其他分享 >聊聊如何利用Testcontainers进行集成测试

聊聊如何利用Testcontainers进行集成测试

时间:2024-06-11 09:34:14浏览次数:24  
标签:集成 testcontainers Testcontainers spring redis 聊聊 docker junit

前言

1、何为Testcontainers?

Testcontainers是一个库,它为引导本地开发和测试依赖关系提供了简单而轻量级的API,并将真实的服务封装在Docker容器中。使用Testcontainers,您可以编写依赖于您在生产中使用的相同服务的测试,而不需要mock或内存服务。

用比较直白的话就是testcontainers 能够让你实现通过编程语言去启动Docker容器,并在程序测试结束后,自动关闭容器

2、Testcontainers有哪些优势?

  • 每个Test Group都能像写单元测试那样细粒度地写集成测试,保证每个集成单元的高测试覆盖率。
  • Test Group间是做到依赖隔离的,也就是说它们不共享任何一个Docker容器。
  • 保证了生产环境和测试环境的一致性,代码部署到线上时不会遇到因为依赖服务接口不兼容而导致的bug 。
  • Test Group可以并行化运行,减少整体测试运行时间。相比较有些 in-memory的依赖服务实现没有实现很好的资源隔离,比如端口,一旦并行化运行就会出现端口冲突 。
  • 得益于Docker,所有测试都可以在本地环境和
    CI/CD环境中运行,测试代码调试和编写就如同写单元测试。
  • 支持市面上主流的语言以及平台,比如java、go、python等

3、使用Testcontainers有哪些注意点

  • Testcontainers基于Docker,所以使用Testcontainers前需要依赖Docker环境。
  • Testcontainers 提供的环境不能应用于生产环境、只能用于测试环境等场景

4、Testcontainers连接docker的策略

Testcontainers在运行时将会尝试按如下顺序使用以下策略连接到 Docker 守护程序:

环境变量:
– DOCKER_HOST
– DOCKER_TLS_VERIFY
– DOCKER_CERT_PATH

每个变量的作用:

  • DOCKER_HOST to set the url to the docker server.
  • DOCKER_CERT_PATH to load the tls certificates from.
  • UseDOCKER_TLS_VERIFY to enable or disable TLS verification.

默认值
DOCKER_HOST=https://localhost:2376
DOCKER_TLS_VERIFY=1
DOCKER_CERT_PATH=~/.docker

我们可以通过环境变量修改以上值,示例

System.setProperty("DOCKER_HOST","tcp://192.168.0.1:2375")

注: 通过程序修改,我们必须确保System.setProperty,在Testcontainers启动容器之前就已经设置,否则无法生效

以上内容可以在官网https://java.testcontainers.org/supported_docker_environment/查到更详细的介绍

下面就以Testcontainers集成redis,并通过junit5进行单元测试为例进行演示

示例

1、项目中pom引入junit5 gav

 <properties>
        <junit-platform.version>1.9.2</junit-platform.version>
        <junit-jupiter.version>5.9.2</junit-jupiter.version>
    </properties>

  <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter-engine</artifactId>
            <version>${junit-jupiter.version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter-api</artifactId>
            <version>${junit-jupiter.version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.junit.vintage</groupId>
            <artifactId>junit-vintage-engine</artifactId>
            <version>${junit-jupiter.version}</version>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.junit.platform</groupId>
            <artifactId>junit-platform-commons</artifactId>
            <version>${junit-platform.version}</version>
            <scope>test</scope>
        </dependency>

注: 如果使用高本版的springboot,则可以直接引入

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
          <dependency>
            <groupId>org.junit.platform</groupId>
            <artifactId>junit-platform-commons</artifactId>
            <version>${junit-platform.version}</version>
            <scope>test</scope>
        </dependency>

即可

2、项目中pom引入testcontainers gav

<properties>     
<testcontainers.version>1.17.3</testcontainers.version>
    </properties>
  <dependency>
            <groupId>org.testcontainers</groupId>
            <artifactId>testcontainers</artifactId>
            <version>${testcontainers.version}</version>
        </dependency>
        <dependency>
            <groupId>org.testcontainers</groupId>
            <artifactId>junit-jupiter</artifactId>
            <version>${testcontainers.version}</version>
            <scope>test</scope>
        </dependency>

当然也需要引入redis客户端 gav,因为这个大家应该都知道,就不介绍了

3、在我们的单元测试中,让testcontainers运行redis容器

示例代码如下

  @Container
    private static GenericContainer<?> redis = new GenericContainer<>(DockerImageName.parse("redis:6.2.6"))
            .withExposedPorts(6379);

上面的代码的意思是创建镜像为redis:6.2.6容器,并将6379端口暴露出来

同时在测试类上,需要添加@Testcontainers(disabledWithoutDocker = true)
注解

@Testcontainers(disabledWithoutDocker = true)
public class RedisTest {
 @Container
    private static GenericContainer<?> redis = new GenericContainer<>(DockerImageName.parse("redis:6.2.6"))
            .withExposedPorts(6379);
}

4、将我们业务程序能和容器集成

private Jedis jedis;

    @BeforeEach
    public void setUp() {

        int port = redis.getMappedPort(6379);
        jedis = new Jedis(redis.getHost(), port);
    }

5、运行单元测试

@Testcontainers(disabledWithoutDocker = true)
public class RedisTest {

    @Container
    private static GenericContainer<?> redis = new GenericContainer<>(DockerImageName.parse("redis:6.2.6"))
            .withExposedPorts(6379);


    private Jedis jedis;

    @BeforeEach
    public void setUp() {

        int port = redis.getMappedPort(6379);
        jedis = new Jedis(redis.getHost(), port);
    }

    @AfterEach
    public void tearDown() {
        if (jedis != null) {
            jedis.close();
        }
    }

    @Test
    public void testRedisConnectionAndSetAndGet() {
        // 测试连接和简单存取
        String key = "testKey";
        String value = "testValue";

        jedis.set(key, value);
        String result = jedis.get(key);

        assert result.equals(value);
    }

我们可以先观察一下docker容器,可以发现redis容器已经成功运行

再观察一下单元测试结果,和我们预期一样

单元测试结束后,我们再看下容器

发现容器已经销毁

上述的例子在官网也有详细教程,可以查看如下链接
https://java.testcontainers.org/quickstart/junit_5_quickstart/

目前我们项目基本都是和springboot集成,接下来我们简单演示一下testcontainers、springboot、redis集成

完整例子如下

@SpringBootTest(classes = TestcontainersApplication.class,webEnvironment = SpringBootTest.WebEnvironment.NONE)
@Testcontainers(disabledWithoutDocker = true)
public class RedisContainerByDynamicPropertySourceTest {

    @Autowired
    private StringRedisTemplate redisTemplate;

    @Container
    private static GenericContainer<?> redis = new GenericContainer<>(DockerImageName.parse("redis:6.2.6"))
            .withExposedPorts(6379);

//    @BeforeEach
//    public void setUp() {
//
//        System.setProperty("spring.redis.host", redis.getHost());
//        System.setProperty("spring.redis.port", redis.getMappedPort(6379).toString());
//    }

    /**
     * Spring TEST 5.2.5才引入DynamicPropertySource
     * @param registry
     */
    @DynamicPropertySource
    private static void registerRedisProperties(DynamicPropertyRegistry registry) {
        registry.add("spring.redis.host", redis::getHost);
        registry.add("spring.redis.port", () -> redis.getMappedPort(6379)
                .toString());
    }

    @Test
    public void testRedisConnectionAndSetAndGet() {
        // 测试连接和简单存取
        String key = "testKey";
        String value = "testValue";
        redisTemplate.opsForValue().set(key, value);
        String result = redisTemplate.opsForValue().get(key);

        assert Objects.equals(result, value);
    }
}

核心的代码是

  @DynamicPropertySource
    private static void registerRedisProperties(DynamicPropertyRegistry registry) {
        registry.add("spring.redis.host", redis::getHost);
        registry.add("spring.redis.port", () -> redis.getMappedPort(6379)
                .toString());
    }

这个注解是spring5.2.5之后才有,当你事先不知道属性的值时,通过@DynamicPropertySource和DynamicPropertyRegistry 搭配可以实现动态属性绑定。详细介绍可以查看spring官网
https://docs.spring.io/spring-framework/reference/testing/testcontext-framework/ctx-management/dynamic-property-sources.html

注: 如果springboot版本比较低,则需要在项目pom引入如下gav,才能使用DynamicPropertySource

  <spring.version>5.2.15.RELEASE</spring.version>
  <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>${spring.version}</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.version}</version>
            <scope>test</scope>
        </dependency>

查看单元测试结果

在使用Testcontainers踩到的坑

注: 因为我window没开启虚拟化,因此没法安装docker desktop。因此我的示例都是连接远程服务器进行测试

因为要连接到远程的docker服务器,因此需要开启2375端口。开启步骤如下

vim /usr/lib/systemd/system/docker.service
将默认的
ExecStart=/usr/bin/dockerd -H unix://var/run/docker.sock \

修改为如下

ExecStart=/usr/bin/dockerd -H fd:// --containerd=/run/containerd/containerd.sock -H tcp://0.0.0.0:2375

直接追加就行,很多网上都是写

ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375 -H unix://var/run/docker.sock \

这么写会报错。修改后,执行

 systemctl daemon-reload
 service docker restart 

通过

ps -ef | grep docker

查看2375端口是否开启

被挖过矿的朋友应该会知道,很多宿主机就是因为公网暴露2375端口,结果被当成矿机。因此可以通过ssh工具创建隧道,通过隧道访问。示例

不过我这边也是因为通过隧道访问,导致后面非常繁琐

开始讲解坑点

坑一:Testcontainers无法连接到远程docker

一开始我是通过

System.setProperty("DOCKER_HOST","tcp://192.168.0.1:2375")

进行设置,因为我设置的点比Testcontainers创建容器的时间晚,因此导致Testcontainers连接的是本地docker,因为我本地没安装docker,导致无法连接上。

我们可以通过在idea上设置

不过有个博主更厉害,他直接通过代码修改。修改代码内容如下

注: 项目的pom需引入如下GAV

<dependency>
            <groupId>com.github.docker-java</groupId>
            <artifactId>docker-java</artifactId>
            <version>3.2.13</version>
        </dependency>
/**
 * testContainer的docker自定义连接策略
 */
@Slf4j
public class MyDockerClientProviderStrategy extends DockerClientProviderStrategy {

    private final DockerClientConfig dockerClientConfig;

    private static final String DOCKER_HOST = "tcp://127.0.0.1:2375";

	/**
	* 初始化的时候配置dockerClientConfig,我们通过docker-java来连接docker
	*/
    public MyDockerClientProviderStrategy() {
        DefaultDockerClientConfig.Builder configBuilder = DefaultDockerClientConfig.createDefaultConfigBuilder();
        configBuilder.withDockerHost(DOCKER_HOST);
        //通过如下配置:关闭RYUK,解决Could not connect to Ryuk at localhost
        System.setProperty("TESTCONTAINERS_RYUK_DISABLED","true");
//		// 开启dockerTLS校验
//        configBuilder.withDockerTlsVerify(true);
//        // 密钥所在文件夹,换到你的项目目录中即可
//        configBuilder.withDockerCertPath("C:\\Users\\Administrator\\Desktop\\docker");

        dockerClientConfig = configBuilder.build();
    }

	/**
	* 这里定义docker连接配置
	*/
    @Override
    public TransportConfig getTransportConfig() {
        return TransportConfig.builder()
                .dockerHost(dockerClientConfig.getDockerHost())
                .sslConfig(dockerClientConfig.getSSLConfig())
                .build();
    }

	/**
	* 对应上面第二个filter,固定返回true即可。
	*/
    @Override
    protected boolean isApplicable() {
        return true;
    }

    @Override
    public String getDescription() {
        return "my-custom-strategy";
    }
}

在src/main/resources创建META-INF/services/org.testcontainers.dockerclient.DockerClientProviderStrategy
文件内容如下

com.github.lybgeek.testcontainers.MyDockerClientProviderStrategy

其实就spi机制。那个博主的文章内容如下,感兴趣的朋友可以看看
https://blog.csdn.net/LHFFFFF/article/details/127117917

坑二:Could not connect to Ryuk at localhost

我也不懂这个是啥,通过官方的issue
https://github.com/testcontainers/testcontainers-java/issues/3609#issuecomment-769615098
设置

TESTCONTAINERS_RYUK_DISABLED=true

禁用RYUK

关了貌似也没啥影响

坑三:Timed out waiting for container port to open (localhost ports: [] should be listening)

一开始我是通过隧道访问,后面发现每次启动,testcontainer创建的容器端口会变。示例


比如是端口32788,再启动会变成32789。后面我就设置一段随机端口的安全组,比如允许30000-40000端口段可以访问。于是问题就暂时解决

总结

本文仅仅只是抛砖引玉,Testcontainers的官网有更多详细的例子,大家感兴趣可以去了解一下
https://testcontainers.com/guides/

demo链接

https://github.com/lyb-geek/springboot-learning/tree/master/springboot-testcontainers

标签:集成,testcontainers,Testcontainers,spring,redis,聊聊,docker,junit
From: https://www.cnblogs.com/linyb-geek/p/18017275

相关文章

  • 使用 .NET 集成 MinIO 实现高效对象存储
    引言https://min.io/在现代软件开发中,存储和管理大量的非结构化数据(如图片、视频和文档)变得越来越重要。对象存储解决方案如AmazonS3已成为主流,但其高昂的成本和对公有云的依赖使得很多开发者寻求开源和自托管的替代方案。MinIO作为一款高性能的开源对象存储系统,以其兼容......
  • 计算机简史第四章 电子时代之集成电路与摩尔定律
    大规模集成电路的到来集成电路:芯片时代的到来1952年,实用的晶体管问世不久,电子行业还盛行电子管之时,一家为石油行业提供地震勘探服务的公司以极其长远的眼光向贝尔实验室买下了专利许可,并斥资数百万美元押注晶体管市场,而它当时的年利润仅有90万,这无疑是一场没有后路的跨界豪赌......
  • ollama gpu 集成测试qwen2 7b 模型
    昨天测试了下基于ollamacpu模式运行的qwen2对于小参数0.5b以及1.5b的速度还行,但是你的cpu核数以及内存还是需要大一些今天测试下基于gpu的qwen27b模型,以下简单说明下安装ollama如果我们申请的机器包含了GPU,ollama的安装cli还是比较方便的,会直接帮助我们进行gpu驱......
  • 【转载】基于 Docker 的 PHP 集成环境 DNMP
    参考https://github.com/yeszao/dnmp?tab=readme-ov-filehttps://learnku.com/articles/19289https://www.awaimai.com/2120.html源码【下载】(由于限制20m上传,删除.git文件夹)环境软件/系统版本说明WindowsWindows10专业版22H219045.4412DockerDes......
  • SpringBoot3集成Knife4j生成接口文档
    导入依赖<dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-openapi3-jakarta-spring-boot-starter</artifactId><version>4.4.0</version></dependency>注意:SpringBoot......
  • VMware ESXi 8.0U2c macOS Unlocker & OEM BIOS 集成网卡驱动 Marvell AQC 网卡定制版
    VMwareESXi8.0U2cmacOSUnlocker&OEMBIOS集成网卡驱动MarvellAQC网卡定制版VMwareESXi8.0U2cmacOSUnlocker&OEMBIOS集成网卡驱动和NVMe驱动(集成驱动版)发布ESXi8.0U2集成驱动版,在个人电脑上运行企业级工作负载请访问原文链接:VMwareESXi8.0U2cmacOS......
  • 深入了解Git:从数据模型到集成IDEA
    Git是现代软件开发中不可或缺的版本控制工具。理解Git的数据模型、暂存区、命令行接口,并将其集成到IDE(如IntelliJIDEA),可以显著提升开发效率。本文将从底层开始,逐步深入Git的各个方面,并介绍如何将其集成到IntelliJIDEA中。目录Git的数据模型暂存区Git的命令行接口将Git集......
  • Spring Boot集成 Geodesy讲解
    目录1Geodesy1.1什么是geodesy1.2操作实践1.2.1pom.xml1.2.2数学公式计算类1.2.3库包调用1.2.4测试1Geodesy1.1什么是geodesy浩瀚的宇宙中,地球是我们赖以生存的家园。自古以来,人类一直对星球上的位置和彼此的距离着迷。无论是航海探险、贸易往来还是科学研究,精确计算......
  • FFmpeg开发笔记(二十八)Linux环境给FFmpeg集成libxvid
    ​XviD是个开源的视频编解码器,它与DivX一同被纳入MPEG-4规范第二部分的视频标准,但DivX并未开源。早期的MP4视频大多采用XviD或者DivX编码,当时的视频格式被称作MPEG-4。现在常见的H.264后来才增补到MPEG-4规范的第十部分,当然如今使用XviD压缩的视频已经不多了。在《FFmpeg开发实战......
  • Matlab实现基于SVM-Adaboost支持向量机结合Adaboost集成学习时间序列预测(股票价格预测
    %加载时间序列数据data=load(‘stock_data.mat’);X=data.X;%特征矩阵y=data.y;%目标向量%划分训练集和测试集train_ratio=0.8;%训练集比例train_size=round(train_ratio*size(X,1));train_X=X(1:train_size,......