Dart 中包含异步函数的测试函数

Mar*_*łek 3 unit-testing dart flutter

我想测试一个调用其他异步函数的函数,但我不知道如何编写它。函数会是这样的:

function(X x, Y y) {
    x.doSomethingAsync().then((result) {
        if (result != null) {
            y.doSomething();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我想模拟 X 和 Y,运行 X,然后验证是否y.doSomething()被调用。然而我不知道如何等待x.doSomethingAsync()完成。我正在考虑在断言之前做一些等待,但这似乎不是可靠的解决方案。
请问有什么帮助吗?:)

Rém*_*let 7

您可以async/await在飞镖中使用。这将大大简化您的功能:

function(DoSomething x,  DoSomething y) async {
  final result = await x.doSomethingAsync();
  if (result != null) {
    y.doSomething();
  }
}
Run Code Online (Sandbox Code Playgroud)

这样,函数在完成之前不会x.doSomething完成。然后,您可以使用async/await带有 async的相同运算符测试您的函数test

你会有这个:

test('test my function', () async {
  await function(x, y);
});
Run Code Online (Sandbox Code Playgroud)

好的,但是我如何测试函数是否被调用?

为此,您可以使用mockito,这是一个用于测试目的的模拟包。

让我们假设您的 x/y 类是:

class DoSomething {
  Future<Object> doSomethingAsync() async {}
  void doSomething() {}
}
Run Code Online (Sandbox Code Playgroud)

然后,您可以通过使用以下方法模拟您的类方法来使用 Mockito:

// Mock class
class MockDoSomething extends Mock implements DoSomething {
}
Run Code Online (Sandbox Code Playgroud)

最后,您可以通过执行以下操作在测试中使用该模拟:

test('test my function', () async {
  final x = new MockDoSomething();
  final y = new MockDoSomething();
  // test return != null
  when(x.doSomethingAsync()).thenReturn(42);
  await function(x, y);

  verifyNever(x.doSomething());
  verify(x.doSomethingAsync()).called(1);
  // y.doSomething must not be called since x.doSomethingAsync returns 42
  verify(y.doSomething()).called(1);
  verifyNever(y.doSomethingAsync());

  // reset mock
  clearInteractions(x);
  clearInteractions(y);

  // test return == null
  when(x.doSomethingAsync()).thenReturn(null);
  await function(x, y);

  verifyNever(x.doSomething());
  verify(x.doSomethingAsync()).called(1);
  // y must not be called this x.doSomethingAsync returns null here
  verifyZeroInteractions(y);
});
Run Code Online (Sandbox Code Playgroud)