zl程序教程

您现在的位置是:首页 >  .Net

当前栏目

ASP.NET Core 单元测试:如何Mock Url.Page()

2023-03-09 22:10:55 时间

 

本文转载自微信公众号「汪宇杰博客」,作者汪宇杰。转载本文请联系汪宇杰博客公众号。

在 ASP.NET Core 中,当你在 UrlHelperExtensions 类上使用扩展方法时,很难在单元测试中编写Mock。因为Moq框架不支持模拟扩展方法。

问题

例如,我的博客代码中使用了 Url.Page() 方法:

  1. var callbackUrl = Url.Page("/Index"nullnull, Request.Scheme); 

但是单元测试中,像这样 Mock 就会爆:

  1. var mockUrlHelper = new Mock<IUrlHelper>(MockBehavior.Strict);mockUrlHelper.Setup(x => x.Page("/Index"nullnull, It.IsAny<string>())).Returns("callbackUrl").Verifiable(); 

爆炸现场

  1. System.NotSupportedException : Unsupported expression: x => x.Page("/Index"nullnull, It.IsAny<string>())    Extension methods (here: UrlHelperExtensions.Page) may not be used in setup / verification expressions. 

解决方法

我们需要 Mock 这个拓展方法调用的底层方法。在本案例中,底层方法是

  1. Microsoft.AspNetCore.Mvc.IUrlHelper.RouteUrl(UrlRouteContext routeContext) 

我是怎么知道的呢?很简单,.NET 都已经开源多少年了,直接看一眼源代码就能知道微软如何单元测试 UrlHelperExtensions。

https://source.dot.net/

从微软的代码里复制两个助手方法

  1. private Mock<IUrlHelper> CreateMockUrlHelper(ActionContext context = null
  2.     context ??= GetActionContextForPage("/Page"); 
  3.  
  4.     var urlHelper = _mockRepository.Create<IUrlHelper>(); 
  5.     urlHelper.SetupGet(h => h.ActionContext) 
  6.         .Returns(context); 
  7.     return urlHelper; 
  8.  
  9. private static ActionContext GetActionContextForPage(string page) 
  10.     return new() 
  11.     { 
  12.         ActionDescriptor = new() 
  13.         { 
  14.             RouteValues = new Dictionary<string, string> 
  15.             { 
  16.                 { "page", page }, 
  17.             } 
  18.         }, 
  19.         RouteData = new() 
  20.         { 
  21.             Values = 
  22.             { 
  23.                 [ "page" ] = page 
  24.             } 
  25.         } 
  26.     }; 

修改我们的单元测试

  1. var mockUrlHelper = CreateMockUrlHelper();mockUrlHelper.Setup(h => h.RouteUrl(It.IsAny<UrlRouteContext>())).Returns("callbackUrl"); 

现在单元测试就能顺利跑过了!

完整的单元测试代码见下方供参考:

  1. [Test] 
  2. public async Task SignOutAAD() 
  3.     _mockOptions.Setup(m => m.Value).Returns(new AuthenticationSettings 
  4.     { 
  5.         Provider = AuthenticationProvider.AzureAD 
  6.     }); 
  7.  
  8.     var mockUrlHelper = CreateMockUrlHelper(); 
  9.     mockUrlHelper.Setup(h => h.RouteUrl(It.IsAny<UrlRouteContext>())) 
  10.         .Returns("callbackUrl"); 
  11.  
  12.     var ctx = new DefaultHttpContext(); 
  13.     var ctl = CreateAuthController(); 
  14.     ctl.ControllerContext = new() { HttpContext = ctx }; 
  15.     ctl.Url = mockUrlHelper.Object; 
  16.  
  17.     var result = await ctl.SignOut(); 
  18.     Assert.IsInstanceOf(typeof(SignOutResult), result);