自动Mock,让编写单元测试更简单
問(wèn)題
由于依賴注入,特別是構(gòu)造函數(shù)注入的廣泛使用,使得編寫(xiě)單元測(cè)試時(shí),需要使用Mock框架(例如Moq)生成測(cè)試類的依賴接口的"模擬"實(shí)現(xiàn),并驗(yàn)證接口是否按預(yù)期使用。
例如eShopOnContainers中的測(cè)試代碼就使用了Moq,實(shí)現(xiàn)如下:
private?readonly?Mock<IMediator>?_mediatorMock; private?readonly?Mock<IOrderQueries>?_orderQueriesMock; private?readonly?Mock<IIdentityService>?_identityServiceMock; private?readonly?Mock<ILogger<OrdersController>>?_loggerMock;public?OrdersWebApiTest() {_mediatorMock?=?new?Mock<IMediator>();_orderQueriesMock?=?new?Mock<IOrderQueries>();_identityServiceMock?=?new?Mock<IIdentityService>();_loggerMock?=?new?Mock<ILogger<OrdersController>>(); }[Fact] public?async?Task?Cancel_order_with_requestId_success() {//Arrange_mediatorMock.Setup(x?=>?x.Send(It.IsAny<IdentifiedCommand<CancelOrderCommand,?bool>>(),?default(CancellationToken))).Returns(Task.FromResult(true));//Actvar?orderController?=?new?OrdersController(_mediatorMock.Object,?_orderQueriesMock.Object,?_identityServiceMock.Object,?_loggerMock.Object);var?actionResult?=?await?orderController.CancelOrderAsync(new?CancelOrderCommand(1),?Guid.NewGuid().ToString())?as?OkResult;//AssertAssert.Equal(actionResult.StatusCode,?(int)System.Net.HttpStatusCode.OK);}可以看到OrdersController依賴了4個(gè)接口,而每個(gè)測(cè)試用例都必須將Mock對(duì)象傳入構(gòu)造函數(shù),一旦更改了OrdersController的構(gòu)造函數(shù)參數(shù),那么你將需要更改大量單元測(cè)試代碼。
如何簡(jiǎn)化這部分工作?
Moq.AutoMocker
Moq.AutoMocker是一款基于Moq的IoC容器,它可以用來(lái)自動(dòng)創(chuàng)建待測(cè)試類的依賴。
引用nuget包Moq.AutoMock之后,上面的測(cè)試代碼可以修改如下:
[Fact] public?async?Task?Cancel_order_with_requestId_success() {//Arrangevar?autoMocker?=?new?AutoMocker();var?orderController?=?autoMocker.CreateInstance<OrdersController>();var?mediatorMock?=?autoMocker.GetMock<IMediator>();mediatorMock.Setup(x?=>?x.Send(It.IsAny<IdentifiedCommand<CancelOrderCommand,?bool>>(),?default(CancellationToken))).Returns(Task.FromResult(true));//Actvar?actionResult?=?await?orderController.CancelOrderAsync(new?CancelOrderCommand(1),?Guid.NewGuid().ToString())?as?OkResult;//AssertAssert.Equal(actionResult.StatusCode,?(int)System.Net.HttpStatusCode.OK); }可以看到,無(wú)需提前生成Mock對(duì)象,autoMocker.CreateInstance<T>即可創(chuàng)建指定類型的實(shí)例。autoMocker.GetMock<T>可以獲得依賴接口的Mock實(shí)例,然后像以前一樣,執(zhí)行Setup或Verify方法。
結(jié)論
Moq.AutoMocker大大減少了在單位測(cè)試中編寫(xiě)重復(fù)Mock代碼的數(shù)量,簡(jiǎn)化了待測(cè)試類的生成。
如果你使用Moq,我強(qiáng)烈推薦它。
如果你覺(jué)得這篇文章對(duì)你有所啟發(fā),請(qǐng)關(guān)注我的個(gè)人公眾號(hào)”My IO“,記住我!
總結(jié)
以上是生活随笔為你收集整理的自动Mock,让编写单元测试更简单的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 本地搭建K8s环境,并配置Ingress
- 下一篇: Net 5.0 快速开发框架 YC.Bo