如何模拟HttpContext.User

时间:2016-10-25 03:13:56

标签: c# asp.net-mvc unit-testing moq httpcontext

我正在开发Asp.net MVC 5项目,我正在尝试设置模拟以在控制器中返回自定义主体。我有搜索并尝试了不同的方法,但没有一个有效。

我有一个BaseController,我的所有控制器都继承自。 BaseController有一个User属性,它在getter中返回HttpContext.User。 HttpContext.user在项目中调用时返回一个值,但在从单元测试项目调用时返回null。

BaseController

public class CustomPrincipal : IPrincipal, ICustomPrincipal
{
    public IIdentity Identity { get; private set; }

    public string UserId { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public bool   IsStoreUser { get; set; }

    public CustomPrincipal(string username)
    {
        this.Identity = new GenericIdentity(username);
    }
}

自定义主体

public class DocumentsController : BaseController
    {
        public ViewResult ViewDocuments()
        {
            var userType = User.IsStoreUser ? UserType.StoreUser : UserType.Corporate;  ***<== User is null when calling from a unit test.***
        }
    }

控制器

[Test]
public void ViewDocuments_WhenCalled_ShouldReturnViewModel()
{
    // Arrange
    var principal = new CustomPrincipal("2038786");
    principal.UserId = "2038786";
    principal.FirstName = "Test";
    principal.LastName = "User";
    principal.IsStoreUser = true;

    var _mockController = new Mock<DocumentsController>(new UnitOfWork(_context)) { CallBase = true };
        _mockController.Setup(u => u.User).Returns(principal);  ***<== Error - "Invalid setup on a non-virtual (overridable in VB) member: u => u.User"***

    // Act
    var result = _controller.ViewDocuments();
}

测试用例

{{1}}

我正在使用nUnit和Moq创建模拟对象,但我不确定我做错了什么。我需要在DocumentControl中模拟User.IsStore的返回,以在我在测试中创建的自定义主体对象中返回IsStore的值。

2 个答案:

答案 0 :(得分:4)

制作模拟http上下文

private class MockHttpContext : HttpContextBase {
    private readonly IPrincipal user;

    public MockHttpContext(IPrincipal principal) {
        this.user = principal;
    }

    public override IPrincipal User {
        get {
            return user;
        }
        set {
            base.User = value;
        }
    }
}

相应地安排测试。

[Test]
public void ViewDocuments_WhenCalled_ShouldReturnViewModel() {
    // Arrange
    var principal = new CustomPrincipal("2038786");
    principal.UserId = "2038786";
    principal.FirstName = "Test";
    principal.LastName = "User";
    principal.IsStoreUser = true;

    var mockUoW = new Mock<IUnitOfWork>();
    //...setup UoW dependency if needed
    var controller = new DocumentsController(mockUoW.Object);
    controller.ControllerContext = new ControllerContext {
        Controller = controller,
        HttpContext = new MockHttpContext(principal)
    };

    // Act
    var result = controller.ViewDocuments();

    //Assert
    //...assertions
}

不要模拟被测系统。嘲笑它的依赖。

答案 1 :(得分:1)

如果你不直接依赖HttpContext,那就容易多了。创建一个IUserProvider界面和一个依赖于HttpContext的实现(例如HttpContextUserProvider),然后在测试中存根IUserProvider

IUserProvider应该通过依赖注入传递给你的控制器。