(译)使用Mockito模拟外部依赖

栏目: 编程工具 · 发布时间: 7年前

内容简介:在某些情况下,单元测试可能依赖于从实时Web服务或数据库获取数据的类。 这是很不方便的,以下给出了几个原因:因此,您可以“模拟”这些外部依赖,而不是依赖于实时Web服务或数据库。模拟允许我们模拟实时Web服务或数据库,并根据情况返回特定结果。一般来说,您可以通过创建类的替代实现来模拟依赖项。您可以手动编写这些替代实现,也可以使用

在某些情况下,单元测试可能依赖于从实时Web服务或数据库获取数据的类。 这是很不方便的,以下给出了几个原因:

  • 调用实时服务或数据库会降低测试执行速度。
  • 如果Web服务或数据库返回意外结果,则目前的测试可能会失败。这被称为“薄片测试”。
  • 使用实时Web服务或数据库很难测试所有可能的成功和失败场景。

因此,您可以“模拟”这些外部依赖,而不是依赖于实时Web服务或数据库。模拟允许我们模拟实时Web服务或数据库,并根据情况返回特定结果。

一般来说,您可以通过创建类的替代实现来模拟依赖项。您可以手动编写这些替代实现,也可以使用 Mockito包 作为快捷方式。

本文演示了使用Mockito软件包进行模拟的基础知识。有关详细信息,请参阅 Mockito软件包文档

路线

  1. 添加 mockito 以及 test 依赖
  2. 创建一个用来测试的方法
  3. 创建一个带有模拟 http.Client 的test文件
  4. 为每种情况写一个test
  5. 运行测试

1.添加mockito依赖

要使用 mockito 包,首先需要将其与dev_dependencies部分中的flutter_test依赖项一起添加到 pubspec.yaml 文件中。

您还将在此示例中使用 http 包,并将在dependencies部分中定义该依赖项。

dependencies:
  http: <newest_version>
dev_dependencies:
  test: <newest_version>
  mockito: <newest_version>

2.创建一个用来测试的方法

在此示例中,您将需要从Internet配方的Fetch数据中对fetchPost函数进行单元测试。要测试此功能,您需要进行两项更改:

  1. 为方法提供http.Client。这允许您基于不同情况提供正确的http.Client。对于Flutter和服务器端的项目,您可以提供http.IOClient。对于浏览器应用程序,您可以提供http.BrowserClient。对于测试,您提供模拟的http.Client。
  2. 使用提供的客户端从Internet获取数据,而不是静态的http.get方法,因为这很难模拟。

该方法现在应该如下所示:

Future<Post> fetchPost(http.Client client) async {
  final response =
      await client.get('https://jsonplaceholder.typicode.com/posts/1');

  if (response.statusCode == 200) {
    // If the call to the server was successful, parse the JSON
    return Post.fromJson(json.decode(response.body));
  } else {
    // If that call was not successful, throw an error.
    throw Exception('Failed to load post');
  }
}

3.创建一个带有模拟的 http.client 的测试文件

接下来,创建一个测试文件以及 MockClient 类。按照单元测试配方简介中的建议,在根 test 文件夹中创建一个名为 fetch_post_test.dart 的文件。

MockClient类实现 http.Client 类。这允许您将 MockClient 传递给 fetchPost 函数,并允许您在每个测试中返回不同的http响应。

// Create a MockClient using the Mock class provided by the Mockito package.
// Create new instances of this class in each test.
class MockClient extends Mock implements http.Client {}

main() {
  // Tests go here
}

4.为每种情况写一个test

如果你考虑下fetchPost函数,它将做两件事中的一件:

Post

因此,你需要针对这两种情况进行测试。您可以使用MockClient类为成功测试返回“Ok”响应,并为不成功的测试返回错误响应。

为此,请使用Mockito提供的when方法。

// Create a MockClient using the Mock class provided by the Mockito package.
// Create new instances of this class in each test.
class MockClient extends Mock implements http.Client {}

main() {
  group('fetchPost', () {
    test('returns a Post if the http call completes successfully', () async {
      final client = MockClient();

      // Use Mockito to return a successful response when it calls the
      // provided http.Client.
      when(client.get('https://jsonplaceholder.typicode.com/posts/1'))
          .thenAnswer((_) async => http.Response('{"title": "Test"}', 200));

      expect(await fetchPost(client), isInstanceOf<Post>());
    });

    test('throws an exception if the http call completes with an error', () {
      final client = MockClient();

      // Use Mockito to return an unsuccessful response when it calls the
      // provided http.Client.
      when(client.get('https://jsonplaceholder.typicode.com/posts/1'))
          .thenAnswer((_) async => http.Response('Not Found', 404));

      expect(fetchPost(client), throwsException);
    });
  });
}

5.运行测试

现在你的fetchPost()方法和测试都就位了,运行测试。

$ dart test/fetch_post_test.dart

您还可以按照单元测试配方简介中的说明的那样,在你喜欢的编辑器中运行测试。

总结

在此示例中,您已经学习了如何使用Mockito来测试依赖于Web服务或数据库的函数或类。这只是对Mockito库和模拟概念的简短介绍。有关更多信息,请参阅 Mockito包 提供的文档。


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

人人都是产品经理

人人都是产品经理

苏杰 / 电子工业出版社 / 2012-6 / 45.00元

本书为《人人都是产品经理》的升级版,是写给“1到3岁的产品经理”的书,适合刚入门的产品经理、产品规划师、需求分析师,以及对做产品感兴趣的学生,用户体验、市场运营、技术部门的朋友们,特别是互联网、软件行业。作为一名“4岁的产品经理”,作者讲述了过去3年的经历与体会,与前辈们的书不同,本书就像你走到作者身边,说“嗨,哥们!晚上有空吃个饭吗,随便聊聊做产品的事吧”,然后作者说“好啊”。 书名叫“......一起来看看 《人人都是产品经理》 这本书的介绍吧!

JS 压缩/解压工具
JS 压缩/解压工具

在线压缩/解压 JS 代码

HEX CMYK 转换工具
HEX CMYK 转换工具

HEX CMYK 互转工具

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具