Как издеваться над HttpContext в ASP.NET MVC с помощью Moq?

101
[TestMethod]
public void Home_Message_Display_Unknown_User_when_coockie_does_not_exist()
{
    var context = new Mock<HttpContextBase>();
    var request = new Mock<HttpRequestBase>();
    context
        .Setup(c => c.Request)
        .Returns(request.Object);
    HomeController controller = new HomeController();

    controller.HttpContext = context; //Here I am getting an error (read only).
    ...
 }

у моего базового контроллера есть переопределение Initialize, которое получает this requestContext. Я пытаюсь передать это, но что-то делаю неправильно.

protected override void Initialize(System.Web.Routing.RequestContext requestContext)
{
    base.Initialize(requestContext);
}

Где я могу получить дополнительную информацию о имитации моего RequestContext и HttpContext с помощью Moq? Я пытаюсь имитировать файлы cookie и общий контекст.

Гео
источник

Ответы:

61

HttpContext доступен только для чтения, но фактически является производным от ControllerContext, который вы можете установить.

 controller.ControllerContext = new ControllerContext( context.Object, new RouteData(), controller );
tvanfosson
источник
Этот сработал для меня, позволив мне установить фиктивный HttpContext на контроллере.
Джоэл Мэлоун
39

Создайте запрос, ответ и поместите их в HttpContext:

HttpRequest httpRequest = new HttpRequest("", "http://mySomething/", "");
StringWriter stringWriter = new StringWriter();
HttpResponse httpResponse = new HttpResponse(stringWriter);
HttpContext httpContextMock = new HttpContext(httpRequest, httpResponse);
0100110010101
источник
Вопрос касается * базовых классов, т.е. HttpRequestBase, а не HttpRequest - я не уверен, зачем оба нужны мне, и еще больше раздражает то, что они «запечатаны». Невозможно установить LogonUserIdentity :(
Крис Кимптон,
Если они упорядочили мою ссылку, это все еще возможно через удаленное взаимодействие, так что это не должно быть проблемой.
0100110010101
1
@ChrisKimpton: В крайнем случае, отражение всегда есть ;-)
Оливер
Это работает при подключении его к контроллеру, например: controller.ControllerContext = new ControllerContext (новый HttpContextWrapper (httpContextMock), new RouteData (), контроллер);
Андреас Вендель
да. вы действительно можете установить .LogonUserIdentity - _request.Setup (n => n.LogonUserIdentity) .Returns ((WindowsIdentity.GetCurrent));
KevinDeus
12

Спасибо пользователю 0100110010101.

Это сработало для меня, и здесь у меня возникла проблема при написании тестового примера для приведенного ниже кода:

 var currentUrl = Request.Url.AbsoluteUri;

А вот строки, которые решили проблему

HomeController controller = new HomeController();
//Mock Request.Url.AbsoluteUri 
HttpRequest httpRequest = new HttpRequest("", "http://mySomething", "");
StringWriter stringWriter = new StringWriter();
HttpResponse httpResponse = new HttpResponse(stringWriter);
HttpContext httpContextMock = new HttpContext(httpRequest, httpResponse);
controller.ControllerContext = new ControllerContext(new HttpContextWrapper(httpContextMock), new RouteData(), controller);

Может быть полезно для других.

Чандан Кумар
источник
Кажется, я не могу использовать тип HttpRequest - это что-то еще?
Винсент Бускарелло
1
Это не полезно , потому что все поля в HttpRequest неизменны
БР
6

Вот как я использовал ControllerContext для передачи поддельного пути к приложению:

[TestClass]
public class ClassTest
{
    private Mock<ControllerContext> mockControllerContext;
    private HomeController sut;

    [TestInitialize]
    public void TestInitialize()
    {
        mockControllerContext = new Mock<ControllerContext>();
        sut = new HomeController();
    }
    [TestCleanup]
    public void TestCleanup()
    {
        sut.Dispose();
        mockControllerContext = null;
    }
    [TestMethod]
    public void Index_Should_Return_Default_View()
    {

        // Expectations
        mockControllerContext.SetupGet(x => x.HttpContext.Request.ApplicationPath)
            .Returns("/foo.com");
        sut.ControllerContext = mockControllerContext.Object;

        // Act
        var failure = sut.Index();

        // Assert
        Assert.IsInstanceOfType(failure, typeof(ViewResult), "Index() did not return expected ViewResult.");
    }
}
Xolartek
источник
1
Зачем вам нужно было передавать поддельный путь приложения?
the_law
Код MVC выполнит его и выдаст нулевое исключение, если его нет.
Джошуа Рамирес
5

Вот пример того, как это можно настроить: Mocking HttpContext HttpRequest и HttpResponse для UnitTests (с использованием Moq)

Обратите внимание на методы расширения, которые действительно помогают упростить использование этих фиктивных классов:

var mockHttpContext = new API_Moq_HttpContext();

var httpContext = mockHttpContext.httpContext();

httpContext.request_Write("<html><body>".line()); 
httpContext.request_Write("   this is a web page".line());  
httpContext.request_Write("</body></html>"); 

return httpContext.request_Read();

Вот пример того, как написать модульный тест с использованием moq, чтобы проверить, что HttpModule работает должным образом: модульный тест для HttpModule с использованием Moq для обертывания HttpRequest

Обновление: этот API изменен на

Динис Крус
источник
Ссылки не работают - пожалуйста, укажите код в своем ответе
Hades