테스트 초기화 메소드의 HttpContext.Current 모의


177

내가 작성한 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 메소드에는 없습니다. 단위 테스트 / 조롱은 나에게 매우 새롭기 때문에 이것이 올바르게 이루어지고 싶습니다. 내 컨트롤러와 Init 메소드에서 호출되는 모든 라이브러리에서 HttpContext를 공유하도록 HttpContext를 모의하는 올바른 방법은 무엇입니까?

답변:


362

HttpContext.CurrentSystem.Web.HttpContext확장하지 않는 의 인스턴스를 반환합니다 System.Web.HttpContextBase. HttpContextBase나중에 HttpContext조롱하기 어려운 문제를 해결 하기 위해 추가되었습니다 . 두 클래스는 기본적으로 관련이 없습니다 (두 클래스 HttpContextWrapper사이의 어댑터로 사용됨).

다행스럽게도 (User) 및을 HttpContext대체하기에 충분할 정도로 가짜 입니다.IPrincipalIIdentity

다음 코드는 콘솔 응용 프로그램에서도 예상대로 실행됩니다.

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

5
@nfplee - 당신이에 빈 문자열을 전달하면 GenericIdentity생성자, IsAuthenticatedfalse를 돌려줍니다
리처드 Szalay

2
이것은 HttpContext에서 캐시를 조롱하는 데 사용될 수 있습니까?
DevDave

1
그렇습니다. 감사!
DevDave

4
@ CiaranG-MVC는 HttpContextBasemock 할 수 있는를 사용합니다 . MVC를 사용하는 경우 게시 한 해결 방법을 사용할 필요가 없습니다. 계속 진행 하면 컨트롤러를 만들기 전에 게시 한 코드를 실행해야 할 수도 있습니다 .
Richard Szalay

35

Test Init 아래에서도 작업이 수행됩니다.

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

내 솔루션의 별도 테스트 프로젝트에서 HTTPContext에 대한 주소를 얻을 수 없습니다. 컨트롤러 상속을 통해 얻을 수 있습니까?
John Peters

1
System.Web테스트 프로젝트에서 언급 했습니까?
PUG

예. 그러나 내 프로젝트는 MVC 프로젝트입니다. System.Web의 MVC 버전에 해당 네임 스페이스의 하위 집합 만 포함되어있을 수 있습니까?
John Peters

2
@ user1522548 (계정을 만들어야 함) Assembly System.Web.dll, v4.0.0.0에는 HTTPContext가 있으며 소스 코드를 확인했습니다.
PUG

내 실수는 System.Web.MVC 및 NOT System.Web에 대한 파일 참조입니다. 당신의 도움을 주셔서 감사합니다.
John Peters

7

나는 이것이 오래된 주제라는 것을 알고 있지만 단위 테스트를 위해 MVC 응용 프로그램을 조롱하는 것은 우리가 매우 정기적으로하는 일입니다.

방금 Visual Studio 2013으로 업그레이드 한 후 Moq 4를 사용하여 MVC 3 응용 프로그램을 조롱하는 경험을 추가하고 싶었습니다. 디버그 모드에서 단위 테스트가 작동하지 않았으며 변수를 들여다 볼 때 HttpContext에 "표현을 평가할 수 없습니다"라는 메시지가 표시되지 않았습니다. .

Visual Studio 2013에 일부 개체를 평가하는 데 문제가 있음이 밝혀졌습니다. 모의 웹 응용 프로그램을 다시 디버깅하려면 도구 => 옵션 => 디버깅 => 일반 설정에서 "관리되는 호환성 모드 사용"을 확인해야합니다.

나는 일반적으로 다음과 같은 것을한다 :

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" } };
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.