Использование Moq для подделки асинхронного метода для unit test

Я тестирую метод для службы, которая вызывает вызов Web API. Использование обычного HttpClient отлично подходит для модульных тестов, если я также запускаю веб-службу (расположенную в другом проекте в решении) локально.

Однако, когда я проверяю свои изменения, сервер сборки не будет иметь доступа к веб-службе, поэтому тесты будут терпеть неудачу.

Я разработал этот способ для своих модульных тестов, создав интерфейс IHttpClient и реализуя версию, которую я использую в своем приложении. Для модульных тестов я делаю издевательскую версию в комплекте с афорированным методом асинхронной почты. Здесь, где я столкнулся с проблемами. Я хочу вернуть OK HttpStatusResult для этого конкретного теста. Для другого подобного теста я верну неудачный результат.

Тест будет работать, но никогда не завершится. Он висит в ожидании. Я новичок в асинхронном программировании, делегатах и ​​сам Moq, и я долго искал SO и Google, изучая новые вещи, но по-прежнему не могу справиться с этой проблемой.

Вот метод, который я пытаюсь проверить:

public async Task<bool> QueueNotificationAsync(IHttpClient client, Email email)
{
    // do stuff
    try
    {
        // The test hangs here, never returning
        HttpResponseMessage response = await client.PostAsync(uri, content);

        // more logic here
    }
    // more stuff
}

Здесь мой метод unit test:

[TestMethod]
public async Task QueueNotificationAsync_Completes_With_ValidEmail()
{
    Email email = new Email()
    {
        FromAddress = "[email protected]",
        ToAddress = "[email protected]",
        CCAddress = "[email protected]",
        BCCAddress = "[email protected]",
        Subject = "Hello",
        Body = "Hello World."
    };
    var mockClient = new Mock<IHttpClient>();
    mockClient.Setup(c => c.PostAsync(
        It.IsAny<Uri>(),
        It.IsAny<HttpContent>()
        )).Returns(() => new Task<HttpResponseMessage>(() => new HttpResponseMessage(System.Net.HttpStatusCode.OK)));

    bool result = await _notificationRequestService.QueueNotificationAsync(mockClient.Object, email);

    Assert.IsTrue(result, "Queue failed.");
}

Что я делаю неправильно?

Благодарим вас за помощь.

Ответ 1

Вы создаете задачу, но никогда ее не запускаете, поэтому она никогда не заканчивается. Однако не просто запустите задачу - вместо этого перейдите к использованию Task.FromResult<TResult>, который даст вам задание, которое уже выполнено:

...
.Returns(Task.FromResult(new HttpResponseMessage(System.Net.HttpStatusCode.OK)));

Обратите внимание, что вы не будете тестировать фактическую асинхронность таким образом - если вы хотите это сделать, вам нужно сделать немного больше работы, чтобы создать Task<T>, который вы можете контролировать более тонко... но что-то на другой день.

Вы также можете подумать об использовании подделки для IHttpClient, а не насмехаться над всем - это действительно зависит от того, как часто вам это нужно.

Ответ 2

Рекомендуйте ответ @Stuart Grassie выше.

var moqCredentialMananger = new Mock<ICredentialManager>();
moqCredentialMananger
                    .Setup(x => x.GetCredentialsAsync(It.IsAny<string>()))
                    .ReturnsAsync(new Credentials() { .. .. .. });

Ответ 3

Использование Mock.Of<...>(...):

var client = Mock.Of<IHttpClient>(c => 
    c.PostAsync(It.IsAny<Uri>(), It.IsAny<HttpContent>()) == Task.FromResult(new HttpResponseMessage(HttpStatusCode.OK))
);