Как смоделировать Запрос на Контроллер в ASP.Net MVC?
У меня есть контроллер в C# с использованием ASP.Net MVC Framework
public class HomeController:Controller{
public ActionResult Index()
{
if (Request.IsAjaxRequest())
{
//do some ajaxy stuff
}
return View("Index");
}
}
Я получил несколько советов по насмешкам и надеялся протестировать код с помощью следующего и RhinoMocks
var mocks = new MockRepository();
var mockedhttpContext = mocks.DynamicMock<HttpContextBase>();
var mockedHttpRequest = mocks.DynamicMock<HttpRequestBase>();
SetupResult.For(mockedhttpContext.Request).Return(mockedHttpRequest);
var controller = new HomeController();
controller.ControllerContext = new ControllerContext(mockedhttpContext, new RouteData(), controller);
var result = controller.Index() as ViewResult;
Assert.AreEqual("About", result.ViewName);
Однако я продолжаю получать эту ошибку:
Исключение System.ArgumentNullException: System.ArgumentNullException: значение не может быть нулевым. Имя параметра: запрос в System.Web.Mvc.AjaxRequestExtensions.IsAjaxRequest(запрос HttpRequestBase)
Так как Request
Объект на контроллере не имеет установщика. Я попытался заставить этот тест работать правильно, используя рекомендованный код из ответа ниже.
Это использовало Moq вместо RhinoMocks, и при использовании Moq я использую следующее для того же теста:
var request = new Mock<HttpRequestBase>();
// Not working - IsAjaxRequest() is static extension method and cannot be mocked
// request.Setup(x => x.IsAjaxRequest()).Returns(true /* or false */);
// use this
request.SetupGet(x => x.Headers["X-Requested-With"]).Returns("XMLHttpRequest");
var context = new Mock<HttpContextBase>();
context.SetupGet(x => x.Request).Returns(request.Object);
var controller = new HomeController(Repository, LoginInfoProvider);
controller.ControllerContext = new ControllerContext(context.Object, new RouteData(), controller);
var result = controller.Index() as ViewResult;
Assert.AreEqual("About", result.ViewName);
но получите следующую ошибку:
Исключение System.ArgumentException: System.ArgumentException: недопустимая установка для неперезаписываемого члена: x => x.Headers["X-Requested-With"] в Moq.Mock.ThrowIfCantOverride(настройка выражения, MethodInfo methodInfo)
Опять же, похоже, что я не могу установить заголовок запроса. Как мне установить это значение, в RhinoMocks или Moq?
10 ответов
Используя Moq:
var request = new Mock<HttpRequestBase>();
// Not working - IsAjaxRequest() is static extension method and cannot be mocked
// request.Setup(x => x.IsAjaxRequest()).Returns(true /* or false */);
// use this
request.SetupGet(x => x.Headers).Returns(
new System.Net.WebHeaderCollection {
{"X-Requested-With", "XMLHttpRequest"}
});
var context = new Mock<HttpContextBase>();
context.SetupGet(x => x.Request).Returns(request.Object);
var controller = new YourController();
controller.ControllerContext = new ControllerContext(context.Object, new RouteData(), controller);
ОБНОВЛЕНО:
издеваться Request.Headers["X-Requested-With"]
или же Request["X-Requested-With"]
вместо Request.IsAjaxRequest()
,
Для любого, кто использует NSubstitute, я смог изменить приведенные выше ответы и сделать что-то вроде этого... (где Details - это имя метода Action на контроллере)
var fakeRequest = Substitute.For<HttpRequestBase>();
var fakeContext = Substitute.For<HttpContextBase>();
fakeRequest.Headers.Returns(new WebHeaderCollection { {"X-Requested-With", "XMLHttpRequest"}});
fakeContext.Request.Returns(fakeRequest);
controller.ControllerContext = new ControllerContext(fakeContext, new RouteData(), controller);
var model = new EntityTypeMaintenanceModel();
var result = controller.Details(model) as PartialViewResult;
Assert.IsNotNull(result);
Assert.AreEqual("EntityType", result.ViewName);
Вот рабочее решение с использованием RhinoMocks. Я основал его на решении Moq, которое я нашел по адресу http://thegrayzone.co.uk/blog/2010/03/mocking-request-isajaxrequest/
public static void MakeAjaxRequest(this Controller controller)
{
MockRepository mocks = new MockRepository();
// Create mocks
var mockedhttpContext = mocks.DynamicMock<HttpContextBase>();
var mockedHttpRequest = mocks.DynamicMock<HttpRequestBase>();
// Set headers to pretend it's an Ajax request
SetupResult.For(mockedHttpRequest.Headers)
.Return(new WebHeaderCollection() {
{"X-Requested-With", "XMLHttpRequest"}
});
// Tell the mocked context to return the mocked request
SetupResult.For(mockedhttpContext.Request).Return(mockedHttpRequest);
mocks.ReplayAll();
// Set controllerContext
controller.ControllerContext = new ControllerContext(mockedhttpContext, new RouteData(), controller);
}
В текущем .NET (v 5):
var controller = new SomeController(); // SomeController that inherits Microsoft.AspNetCore.Mvc.ControllerBase
var httpContext = new DefaultHttpContext(); // DefaultHttpContext class is part of Microsoft.AspNetCore.Http namespace
httpContext.Request.Headers.Add("origin", "0.0.0.1"); // Add your custom headers to request
controller.ControllerContext.HttpContext = httpContext;
Является ли AjaxRequest методом расширения. Таким образом, вы можете сделать это следующим образом, используя Rhino:
protected HttpContextBase BuildHttpContextStub(bool isAjaxRequest)
{
var httpRequestBase = MockRepository.GenerateStub<HttpRequestBase>();
if (isAjaxRequest)
{
httpRequestBase.Stub(r => r["X-Requested-With"]).Return("XMLHttpRequest");
}
var httpContextBase = MockRepository.GenerateStub<HttpContextBase>();
httpContextBase.Stub(c => c.Request).Return(httpRequestBase);
return httpContextBase;
}
// Build controller
....
controller.ControllerContext = new ControllerContext(BuildHttpContextStub(true), new RouteData(), controller);
Похоже, вы ищете это,
var requestMock = new Mock<HttpRequestBase>();
requestMock.SetupGet(rq => rq["Age"]).Returns("2001");
Использование в контроллере:
public ActionResult Index()
{
var age = Request["Age"]; //This will return 2001
}
Вам нужно смоделировать HttpContextBase и поместить его в свойство ControllerContext, например:
controller.ControllerContext =
new ControllerContext(mockedHttpContext, new RouteData(), controller);
Делать IsAjaxRequest()
чтобы вернуть false во время модульного теста, вам нужно настроить заголовки запросов, а также значение сбора запросов в вашем методе тестирования, как указано ниже:
_request.SetupGet(x => x.Headers).Returns(new System.Net.WebHeaderCollection { { "X-Requested-With", "NotAjaxRequest" } });
_request.SetupGet(x=>x["X-Requested-With"]).Returns("NotAjaxRequest");
Причина настройки обоих скрыта в реализации IsAjaxRequest(), которая приведена ниже:
public static bool IsAjaxRequest(this HttpRequestBase request)<br/>
{
if (request == null)
{
throw new ArgumentNullException("request");
}
return ((request["X-Requested-With"] == "XMLHttpRequest") || ((request.Headers != null) && (request.Headers["X-Requested-With"] == "XMLHttpRequest")));
}
Он использует как коллекцию запросов, так и заголовок, поэтому нам нужно создать настройки как для заголовка, так и для коллекции запросов.
это сделает запрос на возврат false, когда это не ajax-запрос. чтобы сделать его верным, вы можете сделать следующее:
_httpContext.SetupGet(x => x.Request["X-Requested-With"]).Returns("XMLHttpRequest");
Я нашел другой способ добавить объект HttpRequestMessage в ваш запрос во время веб-API следующим образом
[Test]
public void TestMethod()
{
var controllerContext = new HttpControllerContext();
var request = new HttpRequestMessage();
request.Headers.Add("TestHeader", "TestHeader");
controllerContext.Request = request;
_controller.ControllerContext = controllerContext;
var result = _controller.YourAPIMethod();
//Your assertion
}
(Немного опоздал на вечеринку, но я выбрал другой маршрут, поэтому решил поделиться)
Чтобы использовать чистый код / насмешливый способ тестирования этого без создания макетов для классов Http, я реализовал IControllerHelper, который имеет метод Initialise, который принимает запрос в качестве параметра, а затем предоставляет свойства, которые я хочу, например:
public interface IControllerHelper
{
void Initialise(HttpRequest request);
string HostAddress { get; }
}
public class ControllerHelper : IControllerHelper
{
private HttpRequest _request;
public void Initialise(HttpRequest request)
{
_request = request;
}
public string HostAddress => _request.GetUri().GetLeftPart(UriPartial.Authority);
}
Затем в моем контроллере я вызываю инициализацию в начале метода:
_controllerHelper.Initialise(Request);
И тогда мой код зависит только от поддельных зависимостей.
return Created(new Uri($"{_controllerHelper.HostName}/api/MyEndpoint/{result.id}"), result);
Для функциональных тестов я просто заменяю iControllerHelper в композиции на замену.