在testing初始化​​方法中模拟HttpContext.Current

我试图添加unit testing到我已经build立的ASP.NET MVC应用程序。 在我的unit testing中,我使用下面的代码:

[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; } } 

这个testing类从具有以下内容的基类inheritance:

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

在这个方法里面,它调用一个库(我无法控制),它试图运行下面的代码:

 HttpContext.Current.User.Identity.IsAuthenticated 

现在你可能会看到这个问题。 我已经设置了假的HttpContext对控制器,但不是在这个基地的Init方法。 unit testing/嘲笑对我来说是非常新的,所以我想确保我得到正确的答案。 什么是正确的方式来模拟出HttpContext,以便在我的控制器和我的Init方法中调用的任何库共享。

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

幸运的是, HttpContext本身已经足够让你replaceIPrincipal (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] ); 

Test Init下面也会做这个工作。

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

我知道这是一个较老的主题,但是嘲笑unit testing的MVC应用程序是我们经常做的事情。

我只是想增加我的经验在升级到Visual Studio 2013之后,使用Moq 4来嘲讽MVC 3应用程序。没有一个unit testing在debugging模式下工作,并且HttpContext在试图查看variables时显示“无法评估expression式” 。

事实certificate,visual studio 2013在评估一些对象方面存在问题。 要debugging模拟Web应用程序再次工作,我必须检查工具=>选项=>debugging=>常规设置中的“使用托pipe兼容模式”。

我通常做这样的事情:

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

如果你的应用程序第三方在内部redirect,那么最好用下面的方法来模拟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" } };