在测试初始方法中模拟HttpContext.Current
我试图将单元测试添加到我构建的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方法中调用的任何库之间共享。
HttpContext.Current
返回System.Web.HttpContext
一个实例,它不扩展System.Web.HttpContextBase
。 稍后添加HttpContextBase
来解决难以模拟的HttpContext
。 这两个类基本上是无关的( HttpContextWrapper
被用作它们之间的适配器)。
幸运的是, HttpContext
本身fakeable刚好够你更换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]
);
Test Init下面也会做这个工作。
[TestInitialize]
public void TestInit()
{
HttpContext.Current = new HttpContext(new HttpRequest(null, "http://tempuri.org", null), new HttpResponse(null));
YourControllerToBeTestedController = GetYourToBeTestedController();
}
我知道这是一个较老的主题,但是嘲笑单元测试的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);
链接地址: http://www.djcxy.com/p/55899.html