测试初始化方法中模拟 HttpContext.Current

问题描述 投票:0回答:5

我正在尝试向我构建的 ASP.NET MVC 应用程序添加单元测试。在我的单元测试中,我使用以下代码:

[TestMethod]
public void IndexAction_Should_Return_View() {
    var controller = new MembershipController();
    controller.SetFakeControllerContext("TestUser");

    ...
}

使用以下助手来模拟控制器上下文:

public static class FakeControllerContext {
    public static HttpContextBase FakeHttpContext(string username) {
        var context = new Mock<HttpContextBase>();

        context.SetupGet(ctx => ctx.Request.IsAuthenticated).Returns(!string.IsNullOrEmpty(username));

        if (!string.IsNullOrEmpty(username))
            context.SetupGet(ctx => ctx.User.Identity).Returns(FakeIdentity.CreateIdentity(username));

        return context.Object;
    }

    public static void SetFakeControllerContext(this Controller controller, string username = null) {
        var httpContext = FakeHttpContext(username);
        var context = new ControllerContext(new RequestContext(httpContext, new RouteData()), controller);
        controller.ControllerContext = context;
    }
}

此测试类继承自具有以下内容的基类:

[TestInitialize]
public void Init() {
    ...
}

在这个方法中,它调用一个库(我无法控制),该库尝试运行以下代码:

HttpContext.Current.User.Identity.IsAuthenticated

现在您可能可以看到问题所在了。我已经针对控制器设置了假 HttpContext,但没有在此基本 Init 方法中设置。单元测试/模拟对我来说非常陌生,所以我想确保我做对了。对我来说,模拟 HttpContext 以便在我的控制器和 Init 方法中调用的任何库之间共享它的正确方法是什么。

c# unit-testing mocking httpcontext
5个回答
406
投票

HttpContext.Current
返回
System.Web.HttpContext
的实例,它不会扩展
System.Web.HttpContextBase
。后来添加了
HttpContextBase
来解决
HttpContext
难以模拟的问题。这两个类基本上是不相关的(
HttpContextWrapper
用作它们之间的适配器)。

幸运的是,

HttpContext
本身是可伪造的,足以让您替换
IPrincipal
(用户)和
IIdentity

即使在控制台应用程序中,以下代码也会按预期运行:

HttpContext.Current = new HttpContext(
    new HttpRequest("", "http://tempuri.org", ""),
    new HttpResponse(new StringWriter())
    );

// User is logged in
HttpContext.Current.User = new GenericPrincipal(
    new GenericIdentity("username"),
    new string[0]
    );

// User is logged out
HttpContext.Current.User = new GenericPrincipal(
    new GenericIdentity(String.Empty),
    new string[0]
    );

42
投票

下面的测试初始化也将完成这项工作。

[TestInitialize]
public void TestInit()
{
  HttpContext.Current = new HttpContext(new HttpRequest(null, "http://tempuri.org", null), new HttpResponse(null));
  YourControllerToBeTestedController = GetYourToBeTestedController();
}

10
投票

我知道这是一个较旧的主题,但是模拟 MVC 应用程序进行单元测试是我们经常做的事情。

我只是想添加升级到 Visual Studio 2013 后使用 Moq 4 模拟 MVC 3 应用程序的经验。没有一个单元测试在调试模式下工作,并且在尝试查看 HttpContext 时显示“无法评估表达式”变量。

结果 Visual Studio 2013 在评估某些对象时出现问题。为了让调试模拟的 Web 应用程序再次工作,我必须在工具 => 选项 => 调试 => 常规设置中检查“使用托管兼容模式”。

我通常会做这样的事情:

public static class FakeHttpContext
{
    public static void SetFakeContext(this Controller controller)
    {

        var httpContext = MakeFakeContext();
        ControllerContext context =
        new ControllerContext(
        new RequestContext(httpContext,
        new RouteData()), controller);
        controller.ControllerContext = context;
    }


    private static HttpContextBase MakeFakeContext()
    {
        var context = new Mock<HttpContextBase>();
        var request = new Mock<HttpRequestBase>();
        var response = new Mock<HttpResponseBase>();
        var session = new Mock<HttpSessionStateBase>();
        var server = new Mock<HttpServerUtilityBase>();
        var user = new Mock<IPrincipal>();
        var identity = new Mock<IIdentity>();

        context.Setup(c=> c.Request).Returns(request.Object);
        context.Setup(c=> c.Response).Returns(response.Object);
        context.Setup(c=> c.Session).Returns(session.Object);
        context.Setup(c=> c.Server).Returns(server.Object);
        context.Setup(c=> c.User).Returns(user.Object);
        user.Setup(c=> c.Identity).Returns(identity.Object);
        identity.Setup(i => i.IsAuthenticated).Returns(true);
        identity.Setup(i => i.Name).Returns("admin");

        return context.Object;
    }


}

并像这样启动上下文

FakeHttpContext.SetFakeContext(moController);

并直接调用控制器中的方法

long lReportStatusID = -1;
var result = moController.CancelReport(lReportStatusID);

4
投票

如果您的应用程序第三方内部重定向,那么最好通过以下方式模拟 HttpContext :

HttpWorkerRequest initWorkerRequest = new SimpleWorkerRequest("","","","",new StringWriter(CultureInfo.InvariantCulture));
System.Web.HttpContext.Current = new HttpContext(initWorkerRequest);
System.Web.HttpContext.Current.Request.Browser = new HttpBrowserCapabilities();
System.Web.HttpContext.Current.Request.Browser.Capabilities = new Dictionary<string, string> { { "requiresPostRedirectionHandling", "false" } };

0
投票

认证管理顾问 (CMC) 称号为希望展示其在管理咨询领域知识和专业知识的个人提供了一个基准。获得认证的个人具有广泛领域的能力,包括人力资源、战略规划、财务、运营、信息技术、营销、领导变革、人际交往能力、客户沟通等等。

© www.soinside.com 2019 - 2024. All rights reserved.