消费者驱动的契约测试 Spring Cloud Contract介绍

什么是契约测试

测试是软件流程中非常重要,不可或缺的一个环节。一般的测试分为单元测试,集成测试,端到端的手工测试,这也是构成测试金字塔的三个层级。我们今天将要讨论的话题是契约测试,它是处于单元测试和集成测试中间的一个环节。这三个层级分别测试的场景如下:

  • 单元测试:测试单个service
  • 集成测试:测试由多个services组成的系统
  • 端到端测试:测试从用户到各个外部系统的整个场景

契约测试的作用:

  • 测试接口和接口之间的正确性
  • 验证服务层提供的数据是否是消费端所需要的
  • 将本来需要在集成测试中体现的问题前移,更早的发现问题
  • 更快速的验证消费端和提供端之间交互的基本正确性

为什么要存在契约测试?

首先我们将使用以下示例模型来描述微服务测试背后的概念:

在上面的图中,我们可以看到有两个微服务,通过REST彼此进行通信。第一项服务扮演消费者的角色,第二项扮演提供者的角色。

当需要进行集成测试时,可以通过服务虚拟化来模拟正在与之通信的微服务。这里服务提供者被模拟,在部署消费者服务之前,您希望证明其能正常工作。当运行所有测试均为绿色您认为可以部署您的服务了。

但是,如果您针对生产提供商运行服务,而不是模拟版本,则有可能会失败。在这个例子中,提供者已经改变了数据格式。集成测试无法解决这个问题,因为它们正在针对Provider的过时版本运行。

如何填补测试过程中的这个空白?

将引入消费者驱动契约测试的概念。消费者驱动契约测试方法是在消费者和提供者之间定义在它们彼此之间转移的数据格式。通常,合同的格式由消费者定义并与相应的提供商共享。之后,执行测试以验证契约是否相符。CDC测试的先决条件之一是可以与提供商服务团队保持良好的最佳密切沟通,分享这些契约和交流测试结果是实施适当的CDC测试的重要部分。

PACT测试框架
PACT是一个开源的CDC测试框架。它提供了广泛的语言支持,如Ruby,Java,Scala,.NET,Javascript,Swift/Objective-C。

PACT的工作原理

消费者作为数据的最终使用者非常清楚、明确的知道需要的什么样格式,什么类型的数据,它将负责创建契约文档(包含结构和格式的json文件),服务提供端将根据消费者端创建的契约文档提供对应格式的数据并返回给消费者,通过契约检查判断如果服务端提供的数据和消费者生成的契约不匹配,将抛出异常并提示给服务提供端。

Spring Cloud Contract
Spring Cloud Contract是一个基于消费者驱动契约的测试框架。它会基于契约来生成存根服务,消费方不需要等待接口开发完成,就可以通过存根服务完成集成测试。Spring Could Contract中,契约是用一种基于 Groovy 的 DSL 定义的。

谈到契约测试时,我们首先需要定义一个包含期望使用接口的第一个文件。作为标准PACT法则,契约必须由消费者服务来定义,但是在Spring Cloud Contract中,它实际上位于提供者服务代码中。在指南手册中包含了两个大步骤:

服务提供者

编写合同规范(Groovy DSL)
在Provider端生成自动验收测试
生成WireMock JSON存根&将存根发布到Maven(本地)存储库

服务消费者

在消费者端配置Stub Runner
执行消费者测试 - Stub Runner嵌入了WireMock
检查验证结果

服务提供者
我们在服务端编写一个简单服务接口,判断数字是奇数还是偶数

@RestController
public class EvenOddController {@GetMapping("/validate/prime-number")public String isNumberPrime(@RequestParam("number") Integer number) {return number % 2 == 0 ? "Even" : "Odd";}
}

MAVEN 依赖
对于我们的提供者,我们需要spring-cloud-starter-contract-verifier依赖:

<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-contract-verifier</artifactId><scope>test</scope>
</dependency>

需要将我们的基础测试类的名称配置到spring-cloud-contract-maven-plugin:

<plugin><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-contract-maven-plugin</artifactId><version>1.2.2.RELEASE</version><extensions>true</extensions><configuration><baseClassForTests>com.peterwanghao.spring.cloud.contract.producer.BaseTestClass</baseClassForTests></configuration>
</plugin>

基础测试类
需要在加载Spring上下文的测试包中添加一个基类:

@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
@DirtiesContext
@AutoConfigureMessageVerifier
public class BaseTestClass {@Autowiredprivate EvenOddController evenOddController;@Beforepublic void setup() {StandaloneMockMvcBuilder standaloneMockMvcBuilder = MockMvcBuilders.standaloneSetup(evenOddController);RestAssuredMockMvc.standaloneSetup(standaloneMockMvcBuilder);}
}

测试存根
在/src/test/ resources/contracts/目录中,我们将在groovy文件中添加测试存根。例如

import org.springframework.cloud.contract.spec.ContractContract.make {description "should return even when number input is even"request {method GET()url("/validate/prime-number") {queryParameters {parameter("number", "2")}}}response {body("Even")status 200}
}

当我们运行构建时,运行 mvn clean install 插件会自动生成一个名为ContractVerifierTest的测试类,它扩展我们的BaseTestClass并将其放在/target/generated-test-sources/contracts/中。

测试方法的名称派生自前缀“ validate_”与我们的Groovy测试存根的名称连接。对于上面的Groovy文件,生成的方法名称将为“validate_shouldReturnEvenWhenRequestParamIsEven”。

我们来看看这个自动生成的测试类:

public class ContractVerifierTest extends BaseTestClass {@Testpublic void validate_shouldReturnEvenWhenRequestParamIsEven() throws Exception {// given:MockMvcRequestSpecification request = given();// when:ResponseOptions response = given().spec(request).queryParam("number","2").get("/validate/prime-number");// then:assertThat(response.statusCode()).isEqualTo(200);// and:String responseBody = response.getBody().asString();assertThat(responseBody).isEqualTo("Even");}
}

构建还将在我们的本地Maven存储库中添加存根jar,以便我们的消费者可以使用它。

服务消费者
我们的CDC消费者将通过HTTP交互生成的存根来维护契约,因此提供者方面的任何更改都将破坏契约。

新建BasicMathController,它将发出HTTP请求以从生成的存根中获取响应:

@RestController
public class BasicMathController {@Autowiredprivate RestTemplate restTemplate;@GetMapping("/calculate")public String checkOddAndEven(@RequestParam("number") Integer number) {HttpHeaders httpHeaders = new HttpHeaders();httpHeaders.add("Content-Type", "application/json");ResponseEntity<String> responseEntity = restTemplate.exchange("http://localhost:8090/validate/prime-number?number=" + number, HttpMethod.GET,new HttpEntity<>(httpHeaders), String.class);return responseEntity.getBody();}
}

MAVEN 依赖
对于我们的消费者,我们需要添加spring-cloud-contract-wiremock和spring-cloud-contract-stub-runner依赖项。还有本地Maven存储库中的可用存根:

<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-contract-wiremock</artifactId><scope>test</scope>
</dependency>
<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-contract-stub-runner</artifactId><scope>test</scope>
</dependency><dependency><groupId>com.peterwanghao.spring.cloud</groupId><artifactId>spring-cloud-contract-producer</artifactId><version>0.0.1-SNAPSHOT</version><scope>test</scope>
</dependency>

存根运行器
现在是时候配置我们的存根运行器,它将通知我们的消费者如何调用我们本地Maven存储库中的可用存根:

@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
@AutoConfigureMockMvc
@AutoConfigureJsonTesters
@AutoConfigureStubRunner(workOffline = true, ids = "com.peterwanghao.spring.cloud:spring-cloud-contract-producer:+:stubs:8090")
public class BasicMathControllerIntegrationTest {@Autowiredprivate MockMvc mockMvc;@Testpublic void given_WhenPassEvenNumberInQueryParam_ThenReturnEven() throws Exception {mockMvc.perform(MockMvcRequestBuilders.get("/calculate?number=2").contentType(MediaType.APPLICATION_JSON)).andExpect(status().isOk()).andExpect(content().string("Even"));}}

通过@AutoConfigureStubRunner自动注入StubRunner,模拟服务方。

参数ids定位到maven中的stub.jar。

Ids = groupId : artifactId : version(’+’表示最新版本): 存根 : StubRunner端口

如果你将stub.jar发布到Maven私服中,可以通过repositoryRoot参数指定私服地址来远程调用。在测试通过后会根据契约返回响应内容。

总结

文中首先介绍了契约测试的背景以及基于CDC开发服务的大致过程。然后编写契约文件通过Spring Cloud Contract的contract verifier插件生成存根和服务提供方的测试用例,消费方编写测试用例,通过StrubRunner模拟服务方来完成一次消费方调用服务方的测试。

转载来源:https://blog.csdn.net/peterwanghao/article/details/86293430 原作者:peterwanghao

消费者驱动的契约测试 Spring Cloud Contract介绍相关推荐

  1. 07 契约测试:如何进行消费者驱动的契约测试?

    上一课时,我讲到了微服务架构下的组件测试,它是针对单个微服务的验收测试,虽然保障了单个微服务功能的正确性,但要想保障微服务间交互功能的正确性,就需要进行契约测试. 契约测试产生的背景 在介绍契约测试之 ...

  2. 消费者驱动的契约测试_告诉我们您想要什么,我们将做到:消费者驱动的合同测试消息传递...

    消费者驱动的契约测试 相当早以前,我们从REST(ful) Web API的角度讨论了消费者驱动的合同测试 ,尤其是将其投射到Java( JAX-RS 2.0规范)的角度. 可以公平地说,至少在公共A ...

  3. 基于消费者驱动的契约测试

    JB Rainsberger 曾说过,"集成测试是一个陷阱,它像一个自我扩散的病毒,无情地威胁着代码库.项目和团队." 随着微服务系统复杂度的增加,集成测试所带来的弊端愈发明显. ...

  4. 契约测试之Spring Cloud Contract

    在微服务架构下,服务间会通过某种形式的消息传递或API调用进行耦合,这让服务的集成以及测试变成了非常具有挑战的一件事.早在微服务流行之前,就有人提出了消费者驱动契约(Consumer-driven c ...

  5. 消费者驱动的微服务契约测试套件:Spring Cloud Contract

    在微服务架构下,你的服务可能由不同的团队提供和维护,在这种情况下,接口的开发和维护可能会带来一些问题,比如服务端调整架构或接口调整而对消费者不透明,导致接口调用失败. 为解决这些问题,Ian Robi ...

  6. 消费者驱动的微服务契约测试套件Spring Cloud Contract

    在微服务架构下,你的服务可能由不同的团队提供和维护,在这种情况下,接口的开发和维护可能会带来一些问题,比如服务端调整架构或接口调整而对消费者不透明,导致接口调用失败. 为解决这些问题,Ian Robi ...

  7. Spring Cloud Contract 契约测试实践

    本文转载公众号:永辉云创技术 该号由我参与维护,欢迎大家关注支持!!! 分布式研发模型演进 众所周知, 分布式系统是由众多微服务构成,并按照功能模块划分后, 由不同的开发小组进行维护. 研发模型如下图 ...

  8. Spring Cloud Contract实践

    1.Spring Cloud Contract简介 Spring Cloud Contract是一个总体项目,其中包含帮助用户成功实施消费者驱动合同方法的解决方案.目前,Spring Cloud Co ...

  9. Spring系列学习之Spring Cloud Contract测试消息传递

    英文原文:https://spring.io/projects/spring-cloud-contract 目录 概述 特性 Spring Boot配置 Server / Producer方面 Cli ...

最新文章

  1. python 函数练习集锦
  2. workerman events.php,workerman安装event扩展的方法介绍
  3. 使用verilog设计实现QR分解
  4. 《深入理解Java函数式编程》系列文章
  5. 第6讲 | 理解区块链之前,先上手体验一把数字货币
  6. 怎么把python添加到path_如何将python的路径加到path中
  7. idea 切换git仓库_Idea切换git分支及合并
  8. Android添加程序背景颜色,Android小程序实现切换背景颜色
  9. 一个VO(Visual Odometry)的简单实现
  10. ai作文批改_每日即评 | 积极适应“AI批改作文”的潮流
  11. 遥控直升机主旋翼设定
  12. magisk卸载内置软件_GJ2x_免root搞机工具箱V6.71 更新:增加卸载内置功能
  13. 一键logo生成器_logo制作助手下载-logo制作助手APP免费版下载v1.1
  14. 智能语音的扫地机器人介简_米家扫地机器人1S评测:升级支持智能语音操控
  15. zabbix监控windows主机网卡流量
  16. inflate方法的用法
  17. 电磁循迹小车赛后总结
  18. Java编程精选题(1)
  19. MATTEO ZANGA网页JS功能实现
  20. 学生用计算机怎么没音效,电脑有声音用播放器没有声音怎么处理啊???

热门文章

  1. 一分钟解决“仅限中国大陆地区播放”
  2. SLAM学习策略和前期准备
  3. const iterator 和const_iterator的区别
  4. Linux(debian7)操作基础(十三)之shell脚本同时执行多个应用程序
  5. Python列出文件夹下某类文件名的方法
  6. Nginx学习之十二-负载均衡-加权轮询策略剖析
  7. 【STL深入学习】SGI STL空间配置器详解(一)-第一级空间配置器
  8. 【C++ STL学习之五】容器set和multiset
  9. ktv 上传图片 1218
  10. 29.课时29.【Django模板】url标签使用详解(Av61533158,P29)