2

나는이 웹 페이지를 찾을 수없는 경우에 슬로우되는 다음과 같은 404 오류 ActionResult :404 오류 MOQ - 보내고

[TestMethod] 
     public void Details_Get_404Handler() 
     { 
      // Arrange 
      var controller = GetController(new Repository(), FakeHttpContext()); 

      // Act 
      var result = controller.Details(3442399) as ViewResult; // invalid Id (not found) 

      //Assert 
      Assert.AreEqual("NotFound", result.ViewName); 
     } 
:

public ActionResult InvokeHttp404(HttpContextBase httpContext) { 
     IController errorController = new ErrorController(); 
     var errorRoute = new RouteData(); 
     errorRoute.Values.Add("controller", "Error"); 
     errorRoute.Values.Add("action", "Http404"); 
     errorRoute.Values.Add("url", httpContext.Request.Url.OriginalString); 
     errorController.Execute(new RequestContext(httpContext, errorRoute)); 
     return new EmptyResult(); 
    } 

그리고 내가 다음 테스트와 단위 테스트를를 시도하고있다

나는 오랫동안이 테스트에 갇혀 있었으며 Url.OriginalString을 요청하는 코드 줄에 null 예외를 던질 것입니다. 다음과 같이 주위를 읽은 후 나는이 이전 게시물 Mocking HttpContextBase with Moq을 발견하고 두 번째 대답은 매우 도움이 발견 :

public static HttpContextBase FakeHttpContext() 
     { 
      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>(); 

      request.SetupGet(x => x.Url).Returns(new Uri("http://localhost/a", UriKind.Absolute)); 
      request.Setup(req => req.ApplicationPath).Returns("~/"); 
      request.Setup(req => req.AppRelativeCurrentExecutionFilePath).Returns("~/"); 
      request.Setup(req => req.PathInfo).Returns(string.Empty); 
      response.Setup(res => res.ApplyAppPathModifier(It.IsAny<string>())) 
       .Returns((string virtualPath) => virtualPath); 
      user.Setup(usr => usr.Identity).Returns(identity.Object); 
      identity.SetupGet(ident => ident.IsAuthenticated).Returns(true); 

      context.Setup(ctx => ctx.Request).Returns(request.Object); 
      context.Setup(ctx => ctx.Response).Returns(response.Object); 
      context.Setup(ctx => ctx.Session).Returns(session.Object); 
      context.Setup(ctx => ctx.Server).Returns(server.Object); 
      context.Setup(ctx => ctx.User).Returns(user.Object); 

      return context.Object; 
     } 

그래서 내가 마지막으로 요청하는 코드의 라인을 구타했다 (I는 URL 문자열 처리했다 라인을 추가) 그 게시물과 대답 덕분에 Url.OriginalString. 하지만 지금은 코드 라인 바로 다음에 붙어 있습니다 : errorController.Execute(new RequestContext(httpContext, errorRoute));. 이제 해당 행에서 널 참조 오류로 테스트가 실패합니다. 나는 이미 httpContext를 위장한 것처럼 여기에 null이 무엇인지에 대해 약간 혼란 스럽다. 누구든지 이걸 도와 줄 수 있니?

편집 :

변경된 ValueProviderFactoriesExtensions : Marnix의 대답에 대한 응답으로 변경

public static class ValueProviderFactoresExtensions 
{ 
    public static ValueProviderFactoryCollection ReplaceNameValueCollectionWith<TOriginal>(this ValueProviderFactoryCollection factories, Func<ControllerContext, NameValueCollection> sourceAccessor) 
    { 
     var original = factories.FirstOrDefault(x => typeof(TOriginal) == x.GetType()); 
     if (original != null) 
     { 
      var index = factories.IndexOf(original); 
      factories[index] = new NameValueCollectionProviderFactory(sourceAccessor); 
     } 
     return factories; 
    } 

    public static ValueProviderFactoryCollection ReplaceHttpFileCollectionWith<TOriginal>(this ValueProviderFactoryCollection factories, Func<ControllerContext, HttpFileCollectionBase> sourceAccessor) 
    { 
     var original = factories.FirstOrDefault(x => typeof(TOriginal) == x.GetType()); 
     if (original != null) 
     { 
      var index = factories.IndexOf(original); 
      factories[index] = new HttpFileCollectionProviderFactory(sourceAccessor); 
     } 
     return factories; 
    } 

    class NameValueCollectionProviderFactory : ValueProviderFactory 
    { 
     private readonly Func<ControllerContext, NameValueCollection> sourceAccessor; 


     public NameValueCollectionProviderFactory(Func<ControllerContext, NameValueCollection> sourceAccessor) 
     { 
      this.sourceAccessor = sourceAccessor; 
     } 


     public override IValueProvider GetValueProvider(ControllerContext controllerContext) 
     { 
      return new NameValueCollectionValueProvider(sourceAccessor(controllerContext), CultureInfo.CurrentCulture); 
     } 

    } 
    class HttpFileCollectionProviderFactory : ValueProviderFactory 
    { 
     private readonly Func<ControllerContext, HttpFileCollectionBase> sourceAccessor; 


     public HttpFileCollectionProviderFactory(Func<ControllerContext, HttpFileCollectionBase> sourceAccessor) 
     { 
      this.sourceAccessor = sourceAccessor; 
     } 


     public override IValueProvider GetValueProvider(ControllerContext controllerContext) 
     { 
      return new HttpFileCollectionValueProvider(controllerContext); 
     } 

    } 
} 

코드 :

아직
public static HttpContextBase FakeHttpContext() 
     { 
      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>(); 
      var files = new Mock<HttpFileCollectionBase>(); 



      request.SetupGet(x => x.Url).Returns(new Uri("http://localhost/a", UriKind.Absolute)); 
      request.Setup(req => req.ApplicationPath).Returns("~/"); 
      request.Setup(req => req.AppRelativeCurrentExecutionFilePath).Returns("~/"); 
      request.Setup(req => req.PathInfo).Returns(string.Empty); 
      request.Setup(req => req.ContentType).Returns("text/html"); 
      request.Setup(req => req.QueryString).Returns(new NameValueCollection()); 
      request.Setup(req => req.Form).Returns(new NameValueCollection()); 
      request.Setup(req => req.Files).Returns(files.Object); 
      response.Setup(res => res.ApplyAppPathModifier(It.IsAny<string>())) 
       .Returns((string virtualPath) => virtualPath); 
      user.Setup(usr => usr.Identity).Returns(identity.Object); 
      identity.SetupGet(ident => ident.IsAuthenticated).Returns(true); 
      context.Setup(ctx => ctx.Request).Returns(request.Object); 
      context.Setup(ctx => ctx.Response).Returns(response.Object); 
      context.Setup(ctx => ctx.Session).Returns(session.Object); 
      context.Setup(ctx => ctx.Server).Returns(server.Object); 
      context.Setup(ctx => ctx.User).Returns(user.Object); 

      ValueProviderFactories.Factories 
       .ReplaceNameValueCollectionWith<FormValueProviderFactory>(ctx => ctx.HttpContext.Request.Form) 
       .ReplaceNameValueCollectionWith<QueryStringValueProviderFactory>(ctx => ctx.HttpContext.Request.QueryString) 
       .ReplaceHttpFileCollectionWith<HttpFileCollectionValueProviderFactory>(ctx => ctx.HttpContext.Request.Files); 

      return context.Object; 
     } 

가 동작하지 않습니다 수는 여전히 null 참조를 던졌습니다 동일한 코드 행. 스택 추적이보기를 찾기 위해 고심하는지 알 수 없습니까? 스택 추적 :

System.Web.Mvc.ViewResult.FindView(ControllerContext context) 
+0

http://stackoverflow.com/a/5229581/293712 – Maheep

+0

내가 오류 컨트롤러가 그 질문에있는 사람처럼하지만 내가 무엇을 물어 보는 것은 Moq를 사용하여 404 오류 컨트롤러를 단위 테스트하는 방법입니다. – DevDave

답변

2

컨트롤러의 컨텍스트를 조롱하는 것은 약간 까다 롭습니다. MVC3에는 값 공급자가 HttpContext.Current를 만지는 것을 방지하기위한 추가 단계가 있습니다.

This answer은 필요한 작업을 설명합니다.

업데이트 : 컨트롤러의 ActionInvoker는 ValueProviderFactories를 직접 사용하지 않을 수 있습니다. 값 공급자의 기본 구현이 웹 요청 외부에서 설정되지 않은 HttpContext.Current에 액세스하여 단위 테스트 중에 NullReferenceException이 발생합니다. 디폴트 값 제공자 (링크 된 응답에 설명 된대로)를 대체하여이를 방지 할 수 있습니다.

복사 테스트 프로젝트에 위의 링크에서 ValueProviderFactoresExtensions 클래스는 FakeHttpContext 방법에 다음 코드를 추가합니다 :

var form = new NameValueCollection(); 
var queryString = new NameValueCollection(); 

request.Setup(x => x.Form).Returns(form); 
request.Setup(x => x.QueryString).Returns(queryString); 

ValueProviderFactories.Factories 
.ReplaceWith<FormValueProviderFactory>(ctx => form) 
.ReplaceWith<QueryStringValueProviderFactory>(ctx => queryString); 
+0

전에 ValueProviderFactories를 사용하지 않았습니까? 질문에 ValueProviderFactoresExtensions 클래스가 있다고 생각할 때 어떻게 테스트 메서드에서 호출 할 수 있는지 보여줄 수있었습니다. 가리키다? 누구든지 Moq에 해결책이있는 것을보고 싶다면 ... – DevDave

+0

ValueProviderFactories를 조정할 방법과 이유를 설명하기 위해 내 대답을 업데이트했습니다. –

+0

테스트 결과 세부 정보의 스택 추적이 이제 namevaluecollections를 지나치고 있지만 지금은 viewname을 찾지 못하는 것 같습니다 ...?내 질문에 사용하고있는 업데이트 된 코드를 게시했습니다. 정확히 말하지 않았지만 동일한 차이점이 있습니까? – DevDave