У меня следующая ошибка 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)