Как макетировать запрос на контроллер в ASP.Net MVC?

171

У меня есть контроллер в 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?

Nissan
источник
Замените Request.IsAjaxRequest на Request.IsAjaxRequest ()
eu-ge-ne
Mock 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.ArgumentException: System.ArgumentException: Неверная настройка для не перезаписываемого элемента: x => x.IsAjaxRequest () в Moq.Mock.ThrowIfCantOverride (настройка выражения, MethodInfo methodInfo)
Nissan
Проблема в том, что IsAjaxRequest () является методом статического расширения и не может быть поддельным - я обновил свой ответ.
Ев-гэ-нэ
должен быть context.SetupGet (x => x.Request) .Returns (request.Object); в приведенном выше коде отсутствует «s» при возврате еще. Кроме того, возникает исключение System.ArgumentException: System.ArgumentException: недопустимая установка для не переопределяемого элемента: x => x.Headers ["X-Requested-With"] в Moq .Mock.ThrowIfCantOverride (настройка выражения, MethodInfo methodInfo) сообщение об ошибке
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);
sjmarsh
источник
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
 }
Dr.Sai
источник
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 controller = новый HomeController (репозиторий, LoginInfoProvider); controller.ControllerContext = новый mockedhttpContext, новый RouteData (), контроллер); var result = controller.Index () as ViewResult; Однако все равно получают то же исключение.
Nissan
Ваша ссылка не работает, но похоже, что следующее работает _request.Setup (o => o.Form) .Returns (new NameValueCollection ());
Vdex
2

Чтобы 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-запрос. чтобы вернуть значение true, вы можете сделать следующее:

_httpContext.SetupGet(x => x.Request["X-Requested-With"]).Returns("XMLHttpRequest");
Шарад Растоги
источник
0

Я нашел другой способ добавить объект HttpRequestMessage в ваш запрос во время Web 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 в составе заменителя.

Stu
источник