我在.Net Core 2.2中有一个Web API,如下所示:
[Authorize]
[Route("api/[controller]")]
[ApiController]
public class SomeController : ControllerBase
{
[HttpPost]
public async Task<string> SomeMethodPost()
{
string returnUrl = $"{this.Request.Scheme}://{this.Request.Host}{this.Request.PathBase}/some/redirect";
//Some Third Part Service Call
return serviceResult;
}
}
我想为单元测试中的控制器操作模拟属性“方案”,“主机”和“ PathBase”。我设法在单元测试方法中编写了以下代码:
var request = new Mock<HttpRequest>(MockBehavior.Strict);
request.Setup(x => x.Scheme).Returns("http");
request.Setup(x => x.Host).Returns(HostString.FromUriComponent("http://localhost:8080"));
request.Setup(x => x.PathBase).Returns(PathString.FromUriComponent("/api"));
var mockHttp = new Mock<ControllerBase>(MockBehavior.Strict);
mockHttp.SetupGet(x => x.Request).Returns(request.Object);
但是,最后一行的模拟抛出异常,因为“ Request
”的“ ControllerBase
”是不可覆盖的。我了解抽象类的非虚拟属性的局限性。有什么解决方法吗?
Moq版本为4.13.0。
答案 0 :(得分:3)
更改方法。不要嘲笑被测对象,在这种情况下,该对象是控制器。
通过Request
访问控制器的HttpContext
,可以在安排测试时进行设置。
例如
//Arrange
var request = new Mock<HttpRequest>();
request.Setup(x => x.Scheme).Returns("http");
request.Setup(x => x.Host).Returns(HostString.FromUriComponent("http://localhost:8080"));
request.Setup(x => x.PathBase).Returns(PathString.FromUriComponent("/api"));
var httpContext = Mock.Of<HttpContext>(_ =>
_.Request == request.Object
);
//Controller needs a controller context
var controllerContext = new ControllerContext() {
HttpContext = httpContext,
};
//assign context to controller
var controller = new SomeController(){
ControllerContext = controllerContext,
};
String expected = "expected value here";
//Act
String actual = await controller.SomeMethodPost();
//Assert
Assert.AreEqual(expected, actual);
//...