dart - 在 Dart 中测试异常
问题描述
当围绕会抛出异常的代码编写测试时,Dart/Mockito(或其他任何东西)如何避免抛出真正的异常?例如,这些测试应该通过并检测到抛出的异常——但 Dart 在第一次测试中抛出了一个真正的异常,所以只有“它接收到一个 Todo”通过。
void main() {
test('It throws an exception', () async {
final client = MockClient();
when(client.get(Uri.parse('https://jsonplaceholder.typicode.com/todos/1'))).thenAnswer((_) async => http.Response('', 404));
expect(await fetchTodo(client, 1), throwsException);
});
test('It receives a Todo', () async {
final client = MockClient();
final jsonString = '''
{
"id": 1,
"userId": 1,
"title": "test",
"completed": false
}
''';
when(client.get(Uri.parse('https://jsonplaceholder.typicode.com/todos/2'))).thenAnswer((_) async => http.Response(jsonString, 200));
expect(await fetchTodo(client, 2), isA<Todo>());
});
}
和模拟的 get 方法(基于 mockito 生成的代码 -@GenerateMocks([http.Client])
在我的测试文件中使用时我得到相同的结果。
class MockClient extends Mock implements http.Client {
Future<http.Response> get(Uri url, {Map<String, String>? headers}) {
return super.noSuchMethod(Invocation.method(#get, [url], {#headers: headers}), returnValue: Future.value(http.Response('', 200))) as Future<http.Response>;
}
}
class Todo {
int id;
int userId;
String title;
bool completed;
Todo(this.id, this.userId, this.title, this.completed);
}
Future<Todo> fetchTodo(http.Client client, int id) async {
final response = await client.get(Uri.parse('https://jsonplaceholder.typicode.com/todos/$id'));
if(response.statusCode == 200) {
return Todo(1, 1, 'Test', true);
}else {
throw Exception('Failed to fetch resource');
}
}
试运行报告:
00:00 +0: It throws an exception
00:00 +0 -1: It throws an exception [E]
Exception: Failed to fetch resource
test/test.dart 49:5 fetchTodo
00:00 +0 -1: It receives a Todo
00:00 +1 -1: Some tests failed.
解决方案
你的问题是你这样做:
expect(await fetchTodo(client, 1), throwsException);
expect()
是一个普通函数,函数参数在函数被调用之前被评估。(Dart 是一种应用顺序语言。)因此,您必须在调用之前等待fetchTodo
完成(并且 before可以尝试匹配)。expect()
expect
throwsException
Matcher
正如throwsA
文档所解释的(这也适用于throwsException
Matcher
),它必须与零参数函数或Future
. 您不需要(也不应该)await
调用fetchTodo
.
此外,由于您正在调用异步函数,因此无法同步检查期望,因此您还需要使用expectLater
而不是expect
:
await expectLater(fetchTodo(client, 1), throwsException);
推荐阅读
- html - 想要通过从页面顶部引入图像来进行滑动显示过渡
- numpy - Numpy 3D 数组维度和切片
- html - 在Angular 组件的多个实例中使用缓存结果
- r - 分组数据的增长率,3组
- vue.js - 仅生产构建错误:Vuetify 无法找到目标数据应用程序
- html - 寻找将所有 HTTP 链接转换为 HTTPS 的脚本或程序
- python - 日语输入法的使用方法
- node.js - 我有一个使用 pm2 在多台服务器上运行的 aws-amplify 的多页网站。我无法让网站让用户保持登录状态
- angular - 如何在nativescript中旋转图像
- c++ - 使用模板的动态特征声明类型