Ошибка 404 - PullRequest
       60

Ошибка 404

2 голосов
/ 30 ноября 2011

У меня следующая ошибка 404 ActionResult, которая выдается, когда веб-страницы не найдены:

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();
    }

И я пытаюсь выполнить модульное тестирование с помощью следующего теста:

[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);
        }

Я застрял в этом тесте долгое время, и в строке кода, запрашивающей Url.OriginalString, было бы пустое исключение.После прочтения я нашел этот предыдущий пост Mocking HttpContextBase с Moq и нашел второй ответ очень полезным, как показано ниже: (я добавил строку, которая позаботилась о строке Url)

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

Итак, я наконец-то побил строку кода, запрашивающую Url.OriginalString, благодаря этому посту и ответу.Но сейчас я застрял в строке кода сразу после нее: errorController.Execute(new RequestContext(httpContext, errorRoute));.Теперь тест не пройден с ошибкой нулевой ссылки в этой строке.Я немного смущен тем, что здесь пусто, поскольку я уже подделал httpContext.Может ли кто-нибудь помочь мне с этим?

Редактировать:

Изменено ValueProviderFactoriesExtensions:

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

    }
}

Код изменен в ответ на ответ Marnix:

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

Все еще не может заставить это работать, все еще бросает нулевую ссылку в той же самой строке кода.Не знаете, пытается ли трассировка стека найти представление?трассировка стека:

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

1 Ответ

2 голосов
/ 30 ноября 2011

Создание контекста для контроллера немного сложно.В MVC3 есть дополнительный шаг, чтобы запретить поставщикам значений касаться HttpContext.Current.

Этот ответ объясняет, что вам нужно сделать.

Обновление: Возможно, вы напрямую не используете ValueProviderFactories, как это делает ActionInvoker на контроллере.Где-то в глубине души реализация по умолчанию для поставщиков значений обращается к 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);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...