如何模拟 HttpContext.User

How to mock HttpContext.User

我正在开发一个 Asp.net MVC 5 项目,我正在尝试为 return 控制器中的自定义主体设置一个模拟。我已经搜索并尝试了建议的不同方法,但其中 none 有效。

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

基础控制器

public class BaseController : Controller
{
    protected virtual new CustomPrincipal User
    {
        get { return HttpContext.User as CustomPrincipal; }  ***<== Line with issue***
    }
}

自定义主体

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();
}

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

制作模拟 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
}

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

如果您不直接依赖 HttpContext,事情会变得容易得多。创建一个 IUserProvider 接口和一个依赖于 HttpContext 的实现(例如 HttpContextUserProvider),然后在你的测试中存根 IUserProvider

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