详解 Rust 如何 Mock HTTP 服务

发表于 2年以前  | 总阅读数:677 次

在 Rust 如何模拟外部 HTTP 服务以进行自动测试和原型设计?

在某些时候,大多数开发人员需要测试与外部 HTTP 服务交互的应用程序,例如第三方 API、身份验证提供程序或数据源。我们并非始终可以使用这些服务,尤其是在自动测试或新服务原型设计期间。为了验证我们的应用程序是否按预期使用这些 API,我们需要一些工具来验证传出的请求,并模仿针对我们的用例和测试场景量身定制的 API 响应。这就是模拟(mocking)库可以提供帮助的地方。

HTTP 模拟库通常允许你创建 HTTP 服务器,并将其配置为自定义请求/响应方案。虽然大多数库都提供了在自动测试中模拟 HTTP 服务器的功能,但有些库还使你能够运行可配置的独立服务器应用程序,该应用程序一次模拟多个应用程序的 API。本文展示了如何使用这些工具来模拟 Rust 中的 HTTP 服务。

01 应用程序

假设我们正在构建一个 Rust 应用程序,它将为我们创建一个 GitHub 存储库。为了执行这些操作,我们将使用 GitHub REST API。然后,我们将编写一些自动测试,通过模拟来自 GitHub 的 HTTP 响应来验证正确的行为。

让我们开始吧

让我们首先为应用程序创建一个新的 cargo 包并命名为:github_api_client

cargo new github_api_client --bin

我们还需要一些库,因此将它们添加到依赖项列表中:

  • isahc作为我们的 HTTP 客户端库
  • serde_json便于 JSON 序列化和反序列化
  • custom_error 用来创建自定义错误类型

Cargo.toml:

[dependencies]
isahc = { version = "1.6", features = ["json"]}
serde_json = "1.0"
anyhow = "1.0"

客户端

现在,让我们添加以下代码,这些代码将允许我们访问 GitHub REST API。我们将创建一个名为 GithubClient 的结构体,该结构将包含访问 GitHub API 所需的所有逻辑。

main.rs:

use isahc::{ReadResponseExt, Request, RequestExt};
use serde_json::{json, Value};
use anyhow::{Result,ensure};

pub struct GithubClient {
    base_url: String,
    token: String,
}

impl GithubClient {
    pub fn new(token: &str, base_url: &str) -> GithubClient {
        GithubClient { base_url: base_url.into(), token: token.into() }
    }

    pub fn create_repo(&self, name: &str) -> Result<String> {
        let mut response = Request::post(format!("{}/user/repos", self.base_url))
            .header("Authorization", format!("token {}", self.token))
            .header("Content-Type", "application/json")
            .body(json!({ "name": name, "private": true }).to_string())?
            .send()?;

        let json_body: Value = response.json()?;

        ensure!(response.status().as_u16() == 201, "Unexpected status code");
        ensure!(json_body["html_url"].is_string(), "Missing html_url in response");

        return Ok(json_body["html_url"].as_str().unwrap().into());
    }
}

fn main() {
    let github = GithubClient::new("<github-token>", "https://api.github.com");
    let url = github.create_repo("myRepo").expect("Cannot create repo");
    println!("Repo URL: {}", url);
}

我们的客户端目前提供的唯一方法是 create_repo。它将存储库名称作为参数,并返回包含存储库 URL 作为字符串值的 Result。为了简化此示例,我们使用 anyhow 进行一般错误处理。

问题

现在我们有了一个功能性应用程序,我们希望编写一些测试,以确保它没有任何明显的问题。

一个棘手的部分是找到一个合适的模拟目标,以便我们可以测试客户在不同情况下的行为。在我们的例子中,HTTP 客户端 Request::post 函数看起来是一个很好的起点,因为它是我们的 GitHub 客户端在向 GitHub API 服务器发送请求之前调用的最后一个函数。

不幸的是,如果没有针对特定 HTTP 客户端的专用模拟库,则模拟 HTTP 客户端不是很实用。这是因为,在更大的应用程序中,我们需要重新实现 HTTP 客户端 API 的很大一部分,以便能够充分模拟请求/响应方案。那么该怎么办呢?

解决方案

为了方便地测试我们的 GitHub API 客户端,我们可以使用 HTTP 模拟库。这些库可以帮助我们实现至少两个测试目标:

  • 验证客户端发送的所有 HTTP 请求是否正确(即包含所有预期值)。
  • 模拟来自 GitHub API 的 HTTP 响应,看看我们的客户是否可以相应地处理它们。

在撰写本文时,至少有以下 Rust 库可以帮助我们做到这一点:

  • mockito
  • httpmock
  • httptest
  • wiremock.

以下表格显示了库之间的比较情况:

Library Execution Custom Matchers Mockable APIs Sync API Async API Stand-alone Mode
mockito serial no 1 yes no no
httpmock parallel yes yes yes yes
httptest parallel yes yes no no
wiremock parallel yes ∞∞ no yes no

根据以上比较,目前提供最完整功能的包是 httpmock。出于这个原因,我们将在本文的其余部分使用这个(并且因为作者是 httpmock 的创建者)。尽管我们将使用 httpmock ,但你也可以在任何其他库中找到大多数类似的功能。

02 创建 Mocks

在本节中,我们将编写一些测试来验证我们的 GitHub API 客户端是否按预期工作。我们首先将 httpmock 添加到我们的依赖项中:

Cargo.toml:

[dev-dependencies]
httpmock = "0.6"

现在我们都准备好了。让我们创建一个测试,以确保客户端实现中的"good path"按预期工作:

main.rs:

// ...
#[cfg(test)]
mod tests {
    use crate::GithubClient;
    use httpmock::MockServer;
    use serde_json::json;

    #[test]
    fn create_repo_success_test() {
        // Arrange
        let server = MockServer::start();
        let mock = server.mock(|when, then| {
            when.method("POST")
                .path("/user/repos")
                .header("Authorization", "token TOKEN")
                .header("Content-Type", "application/json");
            then.status(201)
                .json_body(json!({ "html_url": "http://example.com" }));
        });
        let client = GithubClient::new("TOKEN", &server.base_url());

        // Act
        let result = client.create_repo("myRepo");

        // Assert
        mock.assert();
        assert_eq!(result.is_ok(), true);
        assert_eq!(result.unwrap(), "http://example.com");
    }
}

我们按照 AAA (Arrange-Act-Assert) 模式[1]排列测试代码,该模式将测试分为三个部分:安排,行动和断言。

安排(Arrange)

我们在测试中做的第一件事是创建一个 MockServer 实例(第 11 行),然后我们用它来创建一个 Mock 对象

  1. MockServer 应从传入的 HTTP 请求中应期望的所有值(第 13-16 行)
  2. 如果任何传入的 HTTP 请求与所有预期值匹配,则将发送回 HTTP 响应的规范(第 17-18 行)。

请注意我们如何使用 when 变量来定义请求期望,并使用 then 变量来指定响应值。

模拟服务器只有在收到满足所有期望的 HTTP 请求时才会响应。否则,它将以错误消息和 HTTP 状态代码 404 进行响应。

重要提示:观察我们如何将客户端中的 base URL 设置为指向模拟服务器而不是真正的 GitHub API(第 20 行)。

行动(Act)

在这一部分中,我们执行测试中的代码(第 23 行),即 GithubClientcreate_repo 方法。

断言(Assert)

最后,我们使用 Mock 对象提供的 assert 方法(第 25 行)。此方法可确保模拟服务器只收到一个符合所有期望的 HTTP 请求。否则,它将无法通过测试并打印详细的问题描述(请参阅下一节)。

03 调试

Mock 对象提供了 assert 方法,该方法确保我们的应用程序已向模拟服务器发送了符合所有期望的请求 (即 when)。否则,此方法将无法通过测试。在这种情况下,httpmock 将尝试在其请求日志中找到与你的模拟期望最相似的请求。然后,它将识别两者之间的差异,因此你可以轻松发现不正确或缺失的值。

为了演示此功能,我们将修改客户端以在 content-type 头部发送 text/plain,而不是 application/json。如果我们然后重新运行测试,将看到检测到此更改,并且测试现在失败,并显示以下消息:

At least one request has been received, but none exactly matched the mock specification.
Here is a comparison with the most similar request (request number 1): 
1 : Expected header with name 'Content-Type' and value 'application/json' to be present in the request but it wasn't.
------------------------------------------------------------------------------------------
Expected:                [key=equals, value=equals]   Content-Type=application/json
Actual (closest match):                               content-type=text/plain

根据你的 IDE,还可以在差异查看器中看到预期值和实际值之间的差异。例如,这是它在 IntelliJ 或 CLion 中的样子:

Differences Viewer in IntelliJ and/or CLion.

04 独立模拟服务器

到目前为止,我们只研究了在集成测试中模拟外部 HTTP 服务,这些服务通常在一个应用程序的上下文中运行。另一方面,端到端测试可能涉及多个应用程序。在这种情况下,多个应用程序可能需要访问外部 HTTP 服务,而其中一些应用程序可能不容易用于测试。特别是在早期开发阶段或原型设计期间,某些服务可能仍在开发中,尚未准备好使用。

当多个应用程序需要访问模拟 API 时,在单独的进程中运行专用的模拟服务器,以便为多个应用程序提供模拟 API,这是可行的。这允许测试执行者配置模拟服务器,使其在每个测试场景中都有不同的行为。

Usage of a standalone mock server in end-to-end tests. 一些模拟库附带一个可选的独立模拟服务器。其他的可能有第三方扩展或社区项目,使用独立的模拟服务器扩展库。

例如,httpmock 附带一个单独的 Docker 映像[2],允许你运行专用的模拟服务器。我们可以通过以下方式使用它:

  • 使用 Rust 的动态配置:端到端测试执行器是一个 Rust 测试,其使用方式与之前的 httpmock 集成测试中介绍的方式几乎相同。唯一的区别是它连接到远程模拟服务器(例如,通过使用MockServer::connect[3])而不是创建本地模拟服务器实例(例如,通过使用 MockServer::start[4])。在这种情况下,连接到服务器的所有测试函数都按顺序自动执行以避免冲突。
  • 静态 YAML 文件配置:此模式允许你使用不需要更改的静态配置设置模拟服务器。此方法不需要任何测试执行或 Rust 程序。相反,模拟服务器从 YAML 模拟定义文件中读取其配置,这些文件在结构上与 Rust API 非常相似(请参阅此处的示例[5])。

或者,两种模式混合使用。有关独立模式的更多信息,请参阅 httpmock 文档[6]。

05 总结

本文介绍了如何使用模拟库来模拟 HTTP 服务。我们已经看到了它如何允许我们验证我们的应用程序在自动测试期间发送符合我们期望的 HTTP 请求。我们还可以模拟来自 GitHub API 的 HTTP 响应,以确保我们的应用可以相应地处理它们。此外,还展示了如何在开发过程中使用模拟工具来替换不可用的 HTTP 服务,并使其可以同时访问许多应用程序。

多功能模拟工具在开发生命周期的多个阶段都切实可行,而不仅仅是集成测试。但是,它们对于强化基于 HTTP 的 API 客户端特别有用,并允许我们测试难以重现的边缘情况。

原文链接:https://alexliesenfeld.com/mocking-http-services-in-rust

参考资料

[1]AAA (Arrange-Act-Assert) 模式: https://medium.com/@pjbgf/title-testing-code-ocd-and-the-aaa-pattern-df453975ab80

[2]Docker 映像: https://hub.docker.com/r/alexliesenfeld/httpmock

[3]MockServer::connect: https://docs.rs/httpmock/0.6.4/httpmock/struct.MockServer.html#method.connect

[4]MockServer::start: https://docs.rs/httpmock/0.6.4/httpmock/struct.MockServer.html#method.start

[5]此处的示例: https://github.com/alexliesenfeld/httpmock/blob/master/tests/resources/static_yaml_mock.yaml

[6]httpmock 文档: https://docs.rs/httpmock/0.6.4/httpmock/index.html#standalone-mode

本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/roneeEI2lItudy-eqA07sQ

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:8月以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:8月以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:8月以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:8月以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:8月以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:8月以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:8月以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:8月以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:8月以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:8月以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:8月以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:8月以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:8月以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:8月以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:8月以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:8月以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:8月以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:8月以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:8月以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:8月以前  |  398次阅读  |  详细内容 »
 相关文章
 目录