ASP.NET Core 单元测试
前言
单元测试是好, 但是也很花时间. 有些功能封装好了以后也不怎么会再打开, 所以通常就是徒手测试一下, 过了就过了.
但是往往就是那么神奇, 就是会有需求漏掉. 后来要加, 又由于不想潜水, 对自己有信心就会直接改, 然后只测试自己认为影响的部分. 最后就是有 bug.
所以要解决这种问题, 还是写测试比较好,毕竟一个 live 的项目出 bug 有时候会死的很难看.
3 选 1
其实也没什么好选的, NUnit 是 XUnit 的前生, XUnit 是面向 .NET Core 的.
MSTest 是微软自带的, 但是很少人用.
微软官网三个教程都有. 所以自然是选 XUnit 咯.
文档参考
Unit Testing | .NET Core 101 [7 of 8] (YouTube 教程)
dotnet test (.NET CLI)
dotnet test --filter (.NET CLI)
.NET Core Test Explorer (VS Code Plugin)
Testing .NET Core Apps with Visual Studio Code
大致长相
测试通常是开多一个 Project 做. Naming convention 和 Folder structure 我还没有一个规范, 之后再补上.
IDE 都支持的挺好的. 记得要开启 CodeLens 哦.
VS Code
Visual Studio
.NET CLI
定义 Trait 来做 filter
[Fact]
[Trait("Category", "Now")]
cmd
dotnet test --filter Category=Now
dotnet test --filter Category!=Now
moq
moq 在 TDD 扮演很重要的角色.
参考: moq4 Quickstart
安装: dotnet add package Moq
Service 通常会有许多依赖, 我们在做单侧时不能直接使用这些依赖, 不然依赖有问题就会被误以为是这个 Service 的问题.
所有依赖的 Service 都需要使用 mock 形式.
比如: Options
var options = new RImageOptions
{
Ratios = new List<Ratio> {
new Ratio { Wide = 16, High = 9 },
new Ratio { Wide = 3, High = 2 },
new Ratio { Wide = 4, High = 3 },
new Ratio { Wide = 1, High = 1 },
new Ratio { Wide = 3, High = 4 },
new Ratio { Wide = 2, High = 3 },
}
};
var snapshot = new Mock<IOptionsSnapshot<RImageOptions>>();
snapshot.Setup(s => s.Value).Returns(options);
mainService(snapshot.Object);
常见 Error: "may not be used in setup / verification expressions", 参考这里
原理: Mock<Interface> 它会做一个符合这个 interface 的对象, Mock<Class> 它会 inherit 这个 class 并且 override 它的 method,所以这个 class method 必须是 virtual / abstract 才能被 override
override 失败就报错了。所以呢,最好还是用 Interface。
除了上面这种直接依赖, 也有一种情况是, 我们在测试一个方法 A, 它会执行传入的委托/对象.
而我们想确认它是否真的执行了, 或者执行了几次, 执行时传入的参数是否真确等.
要测试的方法
public static void MatchBracket(string value, string bracket, Action<(int start, int end, string valueInBracket)> action) {
action((0, 2, "a"));
action((4, 6, "b"));
action((8, 14, "c {d}"));
}
检查委托执行顺序, 参数, 调用次数
[Fact]
[Trait("Category", "Now")]
public void MatchBracket()
{
var action = new Mock<Action<(int, int, string)>>(MockBehavior.Strict);
var sequence = new MockSequence();
action.InSequence(sequence).Setup(a => a(Tuple.Create(0, 2, "a").ToValueTuple())); // 确保顺序和参数
action.InSequence(sequence).Setup(a => a(Tuple.Create(4, 6, "b").ToValueTuple()));
action.InSequence(sequence).Setup(a => a(Tuple.Create(8, 14, "c {d}").ToValueTuple()));
Program.MatchBracket("{a} {b} {c {d}}", "{}", action.Object);
action.Verify(a => a(It.IsAny<(int, int, string)>()), Times.Exactly(3)); // 确保调用次数
}
更多 pattern 参考 Quickstart 就可以了. 比如 callbacks 也很好用.
Mocking Extension Methods
它没有优雅的实现方法,只有 workaround。不过视乎大家都不介意。
首先,我们有一个 string extension method ToTitleCase,有一个 Person.SayHello 类和方法,这个方法里面调用了 ToTitleCase。
public static class StringExtensions
{
public static string ToTitleCase(this string stringValue)
{
return stringValue;
}
} public class Person
{
public string SayHello(string name)
{
return $"Hi, {name.ToTitleCase()}";
}
}
现在我们要测试 Person.SayHello,但是又不希望 ToTitleCase 搞鬼,所以需要 mock 这个 ToTitleCase。
测试代码
public class Person_Test
{
[Fact]
public void Do()
{
var person = new Person();
var result = person.SayHello("derrick");
Assert.Equal("Hi, Derrick", result);
}
}
由于 ToTitleCase 有 Bug,所以测试失败了。
但其实我们想测试的是 SayHello,SayHello 的实现并没有 Bug,它是被 ToTitleCase 陷害了。
让我们来 Mock ToTitleCase,首先搞一个中间人。
public interface IStringExtensionsImplement
{
string ToTitleCase(string Value);
} public class StringExtensionsImplement : IStringExtensionsImplement
{
public string ToTitleCase(string stringValue)
{
return stringValue;
}
}
StringExtensionsImplement 负责具体的实现。
而 StringExtensions 只是一个简单的 wrapper 而已。
public static class StringExtensions
{
public static IStringExtensionsImplement Implement { get; set; } = new StringExtensionsImplement(); // 这里让测试可以替换 Mock Implement 进来
public static string ToTitleCase(this string stringValue)
{
return Implement.ToTitleCase(stringValue); // 具体依靠 Implement 对象
}
}
Implement 对象就是中间人。测试的时候我们就可以 Mock 这个对象,然后替换掉 StringExtensions.Implement。
测试代码
public class Person_Test
{
[Fact]
public void Do()
{
var mockImplement = new Mock<IStringExtensionsImplement>();
mockImplement.Setup(e => e.ToTitleCase(It.IsAny<string>())).Returns("Derrick"); // Mock and setup return
var originalImplement = StringExtensions.Implement;
StringExtensions.Implement = mockImplement.Object; // 偷龙转风 var person = new Person();
var result = person.SayHello("derrick");
Assert.Equal("Hi, Derrick", result); StringExtensions.Implement = originalImplement; // 替换回去
}
}
看懂了吗?其原理就是搞一个中间人来做实现,然后 mock 这个中间人再偷龙转风。
ASP.NET Core 单元测试的更多相关文章
- 【转】.NET(C#):浅谈程序集清单资源和RESX资源 关于单元测试的思考--Asp.Net Core单元测试最佳实践 封装自己的dapper lambda扩展-设计篇 编写自己的dapper lambda扩展-使用篇 正确理解CAP定理 Quartz.NET的使用(附源码) 整理自己的.net工具库 GC的前世与今生 Visual Studio Package 插件开发之自动生
[转].NET(C#):浅谈程序集清单资源和RESX资源 目录 程序集清单资源 RESX资源文件 使用ResourceReader和ResourceSet解析二进制资源文件 使用ResourceM ...
- Asp.Net Core 单元测试正确姿势
背景 ASP.NET Core 支持依赖关系注入 (DI) 软件设计模式,并且默认注入了很多服务,具体可以参考 官方文档, 相信只要使用过依赖注入框架的同学,都会对此有不同深入的理解,在此无需赘言. ...
- 关于单元测试的思考--Asp.Net Core单元测试最佳实践
在我们码字过程中,单元测试是必不可少的.但在从业过程中,很多开发者却对单元测试望而却步.有些时候并不是不想写,而是常常会碰到下面这些问题,让开发者放下了码字的脚步: 这个类初始数据太麻烦,你看:new ...
- 使用 xUnit 编写 ASP.NET Core 单元测试
还记得 .NET Framework 的 ASP.NET WebForm 吗?那个年代如果要在 Web 层做单元测试简直就是灾难啊..NET Core 吸取教训,在设计上考虑到了可测试性,就连 ASP ...
- ASP.NET CORE 中用单元测试测试控制器
之前用ASP.NET CORE做的项目 加了一个新功能,数据库加了个字段balabala.... 更新到服务器上,新功能测试正常,然后就没管了..... 今天客户说网站有BUG,某个页面打开后出错了, ...
- K8S+GitLab-自动化分布式部署ASP.NET Core(一) 部署环境
一.部署流程介绍 开发人员通过Git上传asp.net core 项目到Gilab,并编写好.gitlab-ci.yml , GitLab-Runner 自动拉取代码,然后进行Build,编译,单元测 ...
- 使用Visual Studio Code开发Asp.Net Core WebApi学习笔记(九)-- 单元测试
本篇将结合这个系列的例子的基础上演示在Asp.Net Core里如何使用XUnit结合Moq进行单元测试,同时对整个项目进行集成测试. 第一部分.XUnit 修改 Project.json 文件内容, ...
- ASP.NET Core中如何针对一个使用HttpClient对象的类编写单元测试
原文地址: How to unit test a class that consumes an HttpClient with IHttpClientFactory in ASP.NET Core? ...
- 使用 xUnit 编写 ASP.NET Core WebAPI单元测试
本文使用xUnit对ASP.NET Core WebAPI做单元测试,使用HttpClient的同步和异步请求,下面详细介绍xUnit的使用过程: 一.创建示例项目 模板为我们自动创建了一个Value ...
- ASP.NET Core中使用xUnit进行单元测试
单元测试的功能自从MVC的第一个版本诞生的时候,就是作为一个重要的卖点来介绍的,通常在拿MVC与webform比较的时候,单元测试就是必杀底牌,把webform碾压得一无是处. 单元测试的重要性不用多 ...
随机推荐
- 自动化车间3D可视化设计思路
自动化车间3D可视化设计思路 随着国内制造业企业的高速发展,再加上政策支持,高效的生产模式和先进的管理方式越来越受到企业重视.更多的企业将工业信息化技术进行广泛的应用,比如MES系统.数字孪生以及生产 ...
- mysql Using join buffer (Block Nested Loop) join连接查询优化
最近在优化链表查询的时候发现就算链接的表里面不到1w的数据链接查询也需要10多秒,这个速度简直不能忍受 通过EXPLAIN发现,extra中有数据是Using join buffer (Block N ...
- Odoo 基于Win10搭建基于Win10搭建odoo14开发环境搭建
实践环境 win10 Python 3.6.2 odoo_14.0.latest.tar.gz 下载地址: https://download.odoocdn.com/download/14/src?p ...
- Docker Compose在Centos下的安装与卸载
实践环境 Centos7.8 先决条件 已安装Docker Engine 安装Docker Compose 运行以下命令下载稳定版本Docker Compose $ sudo curl -L &quo ...
- onnxruntime无法使用GPU加速 加速失败 解决方法【非常详细】
onnx 无法使用GPU加速 加速失败 解决方法[非常详细] 应该是自目前以来最详细的加速失败解决方法GPU加速,收集了各方的资料.引用资料见后文 硬件配置: GPU CUDA版本:12.2 客户架构 ...
- 【Vue2】Axios、Async+Await、解构赋值
Axios入门使用,Async和Await用法,解构赋值语法 <!DOCTYPE html> <html lang="en"> <head> & ...
- 【Linux】Re04
一.文件所有者 [root@localhost ~]# ls -ahl 总用量 40K dr-xr-x---. 3 root root 187 11月 29 19:20 . dr-xr-xr-x. 1 ...
- 【Mybatis-Plus】Spring整合 驼峰命名设置失效问题
查询时发现这个问题: DEBUG [main] - Creating a new SqlSession DEBUG [main] - SqlSession [org.apache.ibatis.ses ...
- 【Layui】07 徽章 Badge
文档地址: https://www.layui.com/demo/badge.html 圆点徽章: <span class="layui-badge-dot">< ...
- 强化学习算法之DQN算法中的经验池的实现,experience_replay_buffer部分的实现
本文的相关链接: github上DQN代码的环境搭建,及运行(Human-Level Control through Deep Reinforcement Learning)conda配置 ----- ...