Использование 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))
);