c# fake - 在測試初始方法中模擬HttpContext.Current




unit-testing mocking (5)

Test Init下面也會做這個工作。

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

我試圖將單元測試添加到我構建的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:

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

HttpContext.Current返回System.Web.HttpContext一個實例,它不擴展System.Web.HttpContextBase 。 稍後添加HttpContextBase來解決難以模擬的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]
    );

我知道這是一個較老的主題,但是嘲笑單元測試的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);

分別反轉每個組件的位:

Color InvertMeAColour(Color ColourToInvert)
{
   return Color.FromArgb((byte)~ColourToInvert.R, (byte)~ColourToInvert.G, (byte)~ColourToInvert.B);
}

編輯:〜運算符不能自動處理字節,需要強制轉換。





c# unit-testing mocking httpcontext