在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" } };