在测试Init方法中模拟HttpContext.Current


175

我正在尝试将单元测试添加到已构建的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方法中调用的任何库之间共享它的正确方法是什么?

Answers:


361

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

幸运的是,HttpContext它本身就是可伪造的,足以替换IPrincipal(User)和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]
    );

干杯,但是我如何为注销的用户设置呢?
nfplee 2010年

5
@nfplee-如果将空字符串传递给GenericIdentity构造函数,IsAuthenticated则将返回false
Richard Szalay 2010年

2
可以用来模拟HttpContext中的Cache吗?
DevDave

1
是的,可以。谢谢!
DevDave

4
@CiaranG-MVC使用HttpContextBase,可以模拟。如果您使用的是MVC,则无需使用我发布的解决方法。如果继续进行下去,可能甚至需要在创建控制器之前运行我发布的代码。
理查德·萨雷

34

在Test Init下面也可以完成这项工作。

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

我无法在解决方案中的单独测试项目中获得HTTPContext的可寻址性。您能够通过继承控制器来获得它吗?
约翰·彼得斯

1
System.Web在测试项目中有参考吗?
PUG

是的,但是我的项目是MVC项目,System.Web的MVC版本是否可能仅包含该命名空间的子集?
约翰·彼得斯

2
@ user1522548(您应该创建一个帐户)程序集System.Web.dll,v4.0.0.0绝对具有HTTPContext,我刚刚检查了源代码。
PUG 2014年

我的错误,我在文件中有对System.Web.MVC的引用,而不是System.Web的引用。谢谢你的帮助。
约翰·彼得斯

7

我知道这是一门比较老的主题,但是我们非常定期地模拟MVC应用程序进行单元测试。

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

事实证明,Visual Studio 2013在评估某些对象方面存在问题。为了使调试的模拟Web应用程序再次运行,我必须在Tools => Options => Debugging => General设置中检查“ Use Managed Compatibility Mode”。

我通常会这样做:

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

与接受的答案相比,用这种方式进行设置是否有充分的理由?看起来似乎更复杂,并且似乎没有提供任何额外的好处。
kilkfoe

1
它提供了更详细/模块化的模拟方法
Vincent Buscarello

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" } };
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.