无论您是遵循传统的测试金字塔还是采用诸如“测试蜂窝”这样的较新方法,都应该在开发过程中的某个时候开始编写集成测试用例。

您可以编写不同类型的集成测试。从持久性测试开始,您可以检查组件之间的交互,也可以模拟调用外部服务。本文将讨论后一种情况。

在谈论WireMock之前,让我们从一个典型的例子开始。

ChuckNorrisService

我们有一个简单的API,用于手动测试。在“业务”类意外是,它可以调用外部API。它使用Spring 框架提供功能的。没什么特别的。我多次看到的是模拟RestTemplate并返回一些预先确定的答案的测试。该实现可能如下所示:

@Service
public class ChuckNorrisService{
...
public ChuckNorrisFact retrieveFact() {
ResponseEntity<ChuckNorrisFactResponse> response = restTemplate.getForEntity(url, ChuckNorrisFactResponse.class);
return Optional.ofNullable(response.getBody()).map(ChuckNorrisFactResponse::getFact).orElse(BACKUP_FACT);
}
...
}

在检查成功案例的常规单元测试旁边,将至少有一项覆盖HTTP错误码的测试用例,即4xx或5xx状态代码:

@Test
public void shouldReturnBackupFactInCaseOfError() {
String url = "http://localhost:8080";
RestTemplate mockTemplate = mock(RestTemplate.class);
ResponseEntity<ChuckNorrisFactResponse> responseEntity = new ResponseEntity<>(HttpStatus.SERVICE_UNAVAILABLE);
when(mockTemplate.getForEntity(url, ChuckNorrisFactResponse.class)).thenReturn(responseEntity);
ChuckNorrisService service = new ChuckNorrisService(mockTemplate, url);
ChuckNorrisFact retrieved = service.retrieveFact();
assertThat(retrieved).isEqualTo(ChuckNorrisService.BACKUP_FACT);
}

看起来还不错吧?响应实体返回503错误代码,我们的服务不会崩溃。所有测试都是绿色通过的,我们可以部署我们的应用程序。

不幸的是,Spring的RestTemplate不能这样使用。方法签名getForEntity给了我们很小的提示。它指出throws RestClientException。这就是mock的RestTemplate与实际实现不同的地方。我们将永远不会收到ResponseEntity带有4xx或5xx状态代码的。RestTemplate将抛出的子类RestClientException。通过查看类的层次结构,我们可以对可能抛出的结果有一个很好的印象:

因此,让我们看看如何使这项测试更好。

WireMock进行拯救

WireMock通过启动模拟服务器并返回将其配置为返回的答案来模拟Web服务。得益于出色的DSL,它很容易集成到您的测试中,并且模拟请求也很简单。

对于JUnit 4,有一个WireMockRule有助于启动停止服务器的工具。对于JUnit 5,大概需要自己做一个这样的工具。当您检查示例项目时,您可以找到ChuckNorrisServiceIntegrationTest。这是基于JUnit 4的SpringBoot测试。让我们看一下。

最重要的部分是ClassRule:

@ClassRule
public static WireMockRule wireMockRule = new WireMockRule();

如前所述,这将启动和停止WireMock服务器。您也可以像往常一样使用该规则Rule来启动和停止每个测试的服务器。对于我们的测试,这不是必需的。

接下来,您将看到几种configureWireMockFor...方法。这些包含WireMock何时返回答案的说明。将WireMock配置分为几种方法并从测试中调用它们是我使用WireMock的方法。当然,您可以在一个@Before方法中设置所有可能的请求。对于正确使用的Demo,我们这样做:

public void configureWireMockForOkResponse(ChuckNorrisFact fact) throws JsonProcessingException {
ChuckNorrisFactResponse chuckNorrisFactResponse = new ChuckNorrisFactResponse("success", fact);
stubFor(get(urlEqualTo("/jokes/random"))
.willReturn(okJson(OBJECT_MAPPER.writeValueAsString(chuckNorrisFactResponse))));
}

所有方法都是从静态导入的com.github.tomakehurst.wiremock.client.WireMock。如您所见,我们将HTTP GET存入路径/jokes/random并返回JSON对象。该okJson()方法只是带有JSON内容的200响应的简写。对于错误情况,代码甚至更简单:

private void configureWireMockForErrorResponse() {
stubFor(get(urlEqualTo("/jokes/random"))
.willReturn(serverError()));
}

如您所见,DSL使阅读说明变得容易。将WireMock放置在适当的位置,我们可以看到我们先前的实现不起作用,因为RestTemplate引发了异常。因此,我们必须调整代码:

public ChuckNorrisFact retrieveFact() {
try {
ResponseEntity<ChuckNorrisFactResponse> response = restTemplate.getForEntity(url, ChuckNorrisFactResponse.class);
return Optional.ofNullable(response.getBody()).map(ChuckNorrisFactResponse::getFact).orElse(BACKUP_FACT);
} catch (HttpStatusCodeException e){
return BACKUP_FACT;
}
}

这已经涵盖了WireMock的基本用例。配置请求的答案,执行测试,检查结果,so easy,就这么简单。尽管如此,在云环境中运行测试时通常会遇到一个问题。让我们看看我们能做什么。

动态端口上的WireMock

您可能已经注意到,项目中的集成测试包含一个

ApplicationContextInitializer类,并且其@TestPropertySource注释会覆盖实际API的URL。那是因为我想在随机端口上启动WireMock。当然,您可以为WireMock配置一个固定端口,并在测试中将此端口用作常量来处理。但是,如果您的测试在某些云提供商的基础架构上运行,则无法确定该端口是否可用。因此,我认为随机端口更好。

不过,在Spring应用程序中使用属性时,我们必须以某种方式将随机端口传递给我们的服务。或者,如您在示例中看到的那样,覆盖URL。这就是为什么我们使用ApplicationContextInitializer。我们将动态分配的端口添加到应用程序上下文中,然后可以使用属性来引用它${wiremock.port}。这里唯一的缺点是我们现在必须使用ClassRule。否则,我们无法在初始化Spring应用程序之前访问端口。

解决了此问题后,让我们看一下涉及HTTP调用的一个常见问题。

超时时间

WireMock提供了更多的响应可能性,而不仅仅是对GET请求的简单答复。经常被遗忘的另一个测试案例是测试超时。开发人员往往会忘记在RestTemplate设置超时URLConnections。如果没有超时,则两者都将等待无限量的时间来进行响应。在最好的情况下,在最坏的情况下,所有线程都将等待永远不会到达的响应。

因此,我们应该添加一个模拟超时的测试。当然,我们也可以使用Mockito模拟来创建延迟,但是在这种情况下,我们将再次猜测RestTemplate的行为。使用WireMock模拟延迟非常简单:

private void configureWireMockForSlowResponse() throws JsonProcessingException {
ChuckNorrisFactResponse chuckNorrisFactResponse = new ChuckNorrisFactResponse("success", new ChuckNorrisFact(1L, ""));
stubFor(get(urlEqualTo("/jokes/random"))
.willReturn(
okJson(OBJECT_MAPPER.writeValueAsString(chuckNorrisFactResponse))
.withFixedDelay((int) Duration.ofSeconds(10L).toMillis())));
}

withFixedDelay()期望一个表示毫秒的int值。我更喜欢使用Duration或至少一个表示该参数表示毫秒的常量,而不必每次写代码都需要看一下代码注释。

设置超时RestTemplate并添加响应的测试后,我们可以看到RestTemplate抛出ResourceAccessException。因此,我们可以调整catch块以捕获此异常和,HttpStatusCodeException或者仅捕获两者的超类:

public ChuckNorrisFact retrieveFact() {
try {
ResponseEntity<ChuckNorrisFactResponse> response = restTemplate.getForEntity(url, ChuckNorrisFactResponse.class);
return Optional.ofNullable(response.getBody()).map(ChuckNorrisFactResponse::getFact).orElse(BACKUP_FACT);
} catch (RestClientException e){
return BACKUP_FACT;
}
}

现在,我们已经很好地介绍了执行HTTP请求时最常见的情况,并且可以确定我们正在测试接近真实条件的条件。

为什么不?

HTTP集成测试的另一个选择是Hoverfly。它的工作原理类似于WireMock,但我更喜欢后者。原因是在运行包含浏览器的端到端测试时,WireMock也非常有用。Hoverfly(至少是Java库)受JVM代理的限制。这可能使它比WireMock更快,但是当例如某些JavaScript代码开始起作用时,它根本不起作用。当您的浏览器代码也直接调用其他一些服务时,WireMock启动Web服务器这一功能非常有用。然后,您也可以使用WireMock来mock它们,并编写例如Selenium测试。

结论

本文可以向您展示两件事:

  • 集成测试的重要性
  • WireMock是个非常不错的测试框架

当然,这两个主题都可以写出非常多的文章。尽管如此,还是分享了如何使用WireMock及其功能。在以后的学习路上多去阅读他们的文档,然后尝试更多其他功能,例如利用WireMock来进行身份验证。


使用WireMock进行更好的集成测试的更多相关文章

  1. 应用程序框架实战十三:DDD分层架构之我见

    前面介绍了应用程序框架的一个重要组成部分——公共操作类,并提供了一个数据类型转换公共操作类作为示例进行演示.下面准备介绍应用程序框架的另一个重要组成部分,即体系架构支持.你不一定要使用DDD这样的架构 ...

  2. express-4 质量保证(1)

    QA 在Web开发中,质量可以分解为四个维度: 到达率: 到达率是指产品的市场普及程度,即查看网站或使用服务的人数.到达率和盈利能力是正相关关系:访问网站的人越多,购买产品或服务的人就越多.从开发的角 ...

  3. 应用程序框架实战十三:DDD分层架构之我见(转)

    前面介绍了应用程序框架的一个重要组成部分——公共操作类,并提供了一个数据类型转换公共操作类作为示例进行演示.下面准备介绍应用程序框架的另一个重要组成部分,即体系架构支持.你不一定要使用DDD这样的架构 ...

  4. DDD分层架构之我见

    DDD分层架构之我见 前面介绍了应用程序框架的一个重要组成部分——公共操作类,并提供了一个数据类型转换公共操作类作为示例进行演示.下面准备介绍应用程序框架的另一个重要组成部分,即体系架构支持.你不一定 ...

  5. 如何使用TDD和React Testing Library构建健壮的React应用程序

    如何使用TDD和React Testing Library构建健壮的React应用程序 当我开始学习React时,我努力的一件事就是以一种既有用又直观的方式来测试我的web应用程序. 每次我想测试它时 ...

  6. IntelliJ IDEA:Field injection is not recommended

    使用IntelliJ IDEA进行开发的时候,code analyze的时候会出现提示“Field injection is not recommended”. stackoverflow上有篇回答: ...

  7. Java生鲜电商平台-微服务架构概述

    Java生鲜电商平台-微服务架构概述 单体架构存在的问题 在传统的软件技术架构系统中,基本上将业务功能集中在单一应用内,或者是单一进程中.尽管现代化的软件架构理论以及设计原则已推广多年,但实际技术衍化 ...

  8. [Abp vNext 源码分析] - 18. 单元测试

    简介 ABP vNext 框架使用 xUnit 作为单元测试组件,官方的所有模块都编写了大量的 单元/集成测试 确保功能正常.由于 ABP vNext 模块化系统的原因,开发人员在建立单元测试项目的时 ...

  9. ASP.NET Core 中文文档 第五章 测试(5.2)集成测试

    原文: Integration Testing 作者: Steve Smith 翻译: 王健 校对: 孟帅洋(书缘) 集成测试确保应用程序的组件组装在一起时正常工作. ASP.NET Core支持使用 ...

随机推荐

  1. 设计模式C++描述----02.模板(Template)模式

    一. 问题 在面向对象系统的分析与设计过程中经常会遇到这样一种情况:对于某一个业务逻辑(算法实现)在不同的对象中有不同的细节实现,但是逻辑(算法)的框架(或通用的应用算法)是相同的.Template提 ...

  2. 【Linux】【自学笔记】Linux下面docker安装mysql

    写在前面: 捣腾继续,之前把一个SpringBoot的程序安装在docker上面,参考链接:https://www.cnblogs.com/aki-stones/p/2019-11-01-note.h ...

  3. 优化 Git Commit Message

    目前很多项目都是通过 Git 进行管理的,Git 每次提交代码的过程中 提交说明 commit message 是必须的.但仅仅必须是不够的,好的提交说明可以帮助我们提高项目的整体质量. 作用与优点 ...

  4. [考试反思]1023csp-s模拟测试83:等候

    分数倒是依旧那么烂,但是这个时间比较诡异. 6分49秒弄出T1,15分钟送上T2的50分暴力,不到一小时半的时候T3的30分暴力也完成了... 在85分钟之后一次提交也没有 前15分钟平均每分钟得10 ...

  5. Blue:贪心,单调队列

    考场上什么都没想. 显然在扯淡了,应该说是刚开始想了一些没用的. 有决策单调性,所以二分答案? 好,那就二分答案.想想怎么检查每只蛤能不能都跳到终点? 那么每只蛤都不能掉队啊. 如果你现在遇到了一个石 ...

  6. 7.19 NOIP模拟6

    这次考试又一次让mikufun认识到了常数的重要性 T1.那一天我们许下约定 这题一看到D<=1e12,想都没想,矩阵快速幂!然后飞快的码了一个,复杂度n^3logD,让后我观察了一下这个转移矩 ...

  7. 描述Linux发行版的系统目录名称命名规则以及用途

    linux各种发行版都遵循LSB(Linux Stadards Base)规则,使用一致的相关的基础目录名称,使用根目录系统结构(root filesystem),使用FHS(Files Hierar ...

  8. Handler dispatch failed; nested exception is java.lang.NoClassDefFoundError: org/springframework/util/PatternMatchUtils

    { "message": "Handler dispatch failed; nested exception is java.lang.NoClassDefFoundE ...

  9. C语言变量名和地址的关系【转载】//基础的东西

    原文链接:http://blog.csdn.net/ssff1/archive/2009/12/13/4998787.aspx 变量名不占空间 变量:用来标识(identify)一块内存区域,这块区域 ...

  10. 130道ASP.NET面试题(二)

    71.什么是反射?答:动态获取程序集信息 72.用Singleton如何写设计模式答:static属性里面new ,构造函数private 73.什么是Application Pool?答:Web应用 ...