Як знущатися над запитом на контролер у ASP.Net MVC?


171

У мене є контролер в C #, що використовує рамку MVC ASP.Net

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 (налаштування виразів, метод MethodInfoInfo)

Знову здається, я не можу встановити заголовок запиту. Як встановити це значення в RhinoMocks або Moq?


Замініть Request.IsAjaxRequest на Request.IsAjaxRequest ()
eu-ge-ne

Посмішитись Request.Headers ["X-Requested-With"] або Request ["X-Requested-With"]) замість Request.IsAjaxRequest (). Я оновив своє запитання
eu-ge-ne

Відповіді:


212

Використання 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().


1
Я отримую повідомлення "Аргумент типу для методу" ISetupGetter <T, TProperty> Moq.Mock <T> .SetupGet <Tpropert> .... не може бути заражений від uage. Спробуйте точно вказати аргументи типу. Який тип я встановлюю 'var request =' щоб хоч змусити це працювати?
Nissan

Щойно оновив мою відповідь - не Request.IsAjaxRequest, але Request.IsAjaxRequest (). Оновіть своє запитання також
eu-ge-ne

Досі генерує: System Exception System.ArgumentException: System.ArgumentException: Неправильне налаштування для неперевіруемого члена: x => x.IsAjaxRequest () у Moq.Mock.ThrowIfCantOverride (налаштування виразів, methodInfo methodInfo)
Nissan

Проблема полягає в тому, що IsAjaxRequest () - метод статичного розширення, і його не можна глузувати - я оновив свою відповідь.
eu-ge-ne

має бути контекстом.SetupGet (x => x.Request) .Вороти (запит.Об'єкт); у вашому коді вище відсутнє значення 's' при поверненні, все ще призводить до винятку System.ArgumentException: System.ArgumentException: Недійсна установка для неперезаписуваного члена: x => x.Headers ["X-Requested-With"] у Moq .Mock.ThrowIfCantOverride (налаштування виразів, метод MethodInfoInfo) повідомлення про помилку
Nissan

17

Для всіх, хто використовує 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);

13

Ось робоче рішення за допомогою 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);
}

5

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

4

Схоже, ви шукаєте це,

 var requestMock = new Mock<HttpRequestBase>();
 requestMock.SetupGet(rq => rq["Age"]).Returns("2001");

Використання в контролері:

 public ActionResult Index()
 {
        var age = Request["Age"]; //This will return 2001
 }

3

Вам потрібно знущатися над HttpContextBase і помістити його у свою властивість ControllerContext, наприклад:

controller.ControllerContext = 
new ControllerContext(mockedHttpContext, new RouteData(), controller);

і що б знущався над HttpContext? Для цього об'єкта RequestContext потрібен об'єкт HttpContextBase () в конструкторі, а HttpContextBase () не має конструктора, який приймає нульові параметри.
Nissan

Я спробував: var mocks = new MockRepository (); var mockedhttpContext = mocks.DynamicMock <HttpContextBase> (); var mockedHttpRequest = mocks.DynamicMock <HttpRequestBase> (); SetupResult.For (mockedhttpContext.Request) .Повернення (mockedHttpRequest); контролер var = новий HomeController (сховище, LoginInfoProvider); controller.ControllerContext = новий mockedhttpContext, новий RouteData (), контролер); var result = controller.Index () як ViewResult; Однак все одно викинути той самий виняток.
Ніссан

Посилання не працює, але, здається, працює _request.Setup (o => o.Form) .Повернення (новий NameValueCollection ());
Vdex

2

Щоб IsAjaxRequest()повернути помилку під час тестування підрозділу, вам потрібно встановити Запит заголовків, а також значення збору запитів у вашому методі тестування, як зазначено нижче:

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

Він використовує як Збір запиту, так і заголовок. Ось чому нам потрібно створити налаштування як для заголовка, так і для запиту колекції.

це зробить запит на повернення помилковим, якщо це не запит Ajax. щоб зробити його справжнім, ви можете зробити наступне:

_httpContext.SetupGet(x => x.Request["X-Requested-With"]).Returns("XMLHttpRequest");

0

Я знайшов інший спосіб додати об’єкт 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
}

0

(Трохи запізнився на вечірку, але я пішов іншим маршрутом, тому подумав, що поділюсь)

Щоб скористатися чистим кодом / глузливим способом тестування цього без створення макетів для 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 у складі замінника.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.