5

我一直在研究新的 API 包装器,并且不想在每次运行单元测试时都调用 API。因此,如此处所述,我在嘲笑它。

我最初认为我嘲笑它的方式有问题,但似乎问题出在其他地方。

我想要完成的事情非常简单。当我的单元测试运行时,我想返回一个值,就好像我已经出去从我正在集成的外部 API 中获取信息一样。

我使用 http.Client 作为可选参数来初始化我的类,因此我可以在单元测试运行时将其传入:

SampleClass(String arg1, String arg2, [http.Client httpClient = null]) {
    this._arg1 = arg1;
    this._arg2 = arg2;
    _httpClient = (httpClient == null) ? http.Request : httpClient;
}

Future apiRequest(String resource, [Map<String, String> body]) {
    var url = buildBaseUrl(resource).toString();
    var request = new http.Request('POST', Uri.parse(url));
    request.bodyFields = body;
    return this._httpClient.send(request).then((response) => response.stream.bytesToString().then((value) => value.toString()));
}

在我的单元测试中,我创建了以下模拟类:

class HttpClientMock extends Mock implements http.Client {
  noSuchMethod(i) => super.noSuchMethod(i);
}

class HttpResponseMock extends Mock implements http.Response {
    noSuchMethod(i) => super.noSuchMethod(i);
}

在我的单元测试中检查响应我正在执行以下操作:

test("Send SMS errors with wrong account", () {
    var mockHttpClient = new HttpClientMock()
                             ..when(callsTo('send')).alwaysReturn(message401);
    var sample = new SampleClass(_arg1, _arg2, mockHttpClient);
    future = sample.apiRequest(...parameters here...).then((value) => value.toString());
    expect(future.then((value) => JSON.decode(value)), completion(equals(JSON.decode(message401))));
});

所以,正如你所看到的,我正在努力做到这一点,所以调用 send 返回message401,这只是一个JSON字符串。

这没有发生,因为message401它是一个字符串,并且因为我的代码试图将它用作 Future,所以我总是得到错误:

顶级未捕获错误:类“String”没有实例方法“then”。

我完全理解为什么会出现此错误,但不知道如何解决它。

任何帮助表示赞赏。

4

4 回答 4

12

http软件包有一个测试库,其中已经为您实现了MockClient

于 2014-06-14T22:15:28.483 回答
2

尝试

.alwaysReturn(new Future.value(message401));
于 2014-06-14T21:50:08.610 回答
1

有一个nock包:

import 'package:test/test.dart';
import 'package:http/http.dart' as http;
import 'package:nock/nock.dart';

void main() {
  setUpAll(() {
    nock.init();
  });

  setUp(() {
    nock.cleanAll();
  });

  test("example", () async {
    final interceptor = nock("http://localhost/api").get("/users")
      ..reply(
        200,
        "result",
      );

    final response = await http.get("http://localhost/api/users");

    expect(interceptor.isDone, true);
    expect(response.statusCode, 200);
    expect(response.body, "result");
  });
}

它使用 HttpOverrides,所以你不需要注入 MockClient。适用于diohttp包。

于 2020-10-09T14:03:49.860 回答
1

您可以从 pub.dev 获得的包中包含的 MockClient 测试的最小示例http

http 包添加到您的 pubspec.yaml 文件中...

dependencies:
  http: ^0.12.2

在您的单元测试飞镖文件中...

import 'dart:convert';

import 'package:http/http.dart';
import 'package:http/testing.dart';
import 'package:test/test.dart';

void main() {

  test('handles a request', () async {
    var client = MockClient((request) async {
      return Response(json.encode(request.bodyFields), 200, request: request);
    }
    );

    var response = await client.post('http://example.com/foo', body: {'field1': 'value1'});

    expect(response.body, contains('value1'));
  });
}
于 2020-09-19T23:45:45.610 回答