使用Moq模拟单元测试的异步方法

时间:2013-12-31 15:41:23

标签: c# unit-testing asynchronous moq

我正在测试一个进行Web API调用的服务的方法。如果我还在本地运行Web服务(位于解决方案中的另一个项目中),则使用普通HttpClient可以正常进行单元测试。

但是,当我签入我的更改时,构建服务器将无法访问Web服务,因此测试将失败。

通过创建IHttpClient接口并实现我在应用程序中使用的版本,我已经为我的单元测试设计了一种解决方法。对于单元测试,我使用模拟的异步post方法创建一个模拟版本。这是我遇到问题的地方。我想为此特定测试返回OK HttpStatusResult。对于另一个类似的测试,我将返回一个糟糕的结果。

测试将运行但永远不会完成。它挂在等待。我是异步编程,代表和Moq本身的新手,我一直在搜索SO和谷歌一段时间学习新东西,但我似乎仍然无法解决这个问题。

以下是我要测试的方法:

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
}

这是我的单元测试方法:

[TestMethod]
public async Task QueueNotificationAsync_Completes_With_ValidEmail()
{
    Email email = new Email()
    {
        FromAddress = "bob@example.com",
        ToAddress = "bill@example.com",
        CCAddress = "brian@example.com",
        BCCAddress = "ben@example.com",
        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.");
}

我做错了什么?

感谢您的帮助。

3 个答案:

答案 0 :(得分:301)

你正在创建一个任务但从未启动它,所以它永远不会完成。但是,不要只是启动任务 - 而是更改为使用Task.FromResult<TResult>,这将为您提供已完成的任务:

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

请注意,您不会以这种方式测试实际的异步 - 如果您想这样做,您需要做更多的工作来创建一个Task<T>,您可以更精细地控制它方式......但那是另一天的事情。

您可能还想考虑使用假货IHttpClient而不是嘲笑一切 - 这实际上取决于您需要它的频率。

答案 1 :(得分:9)

推荐上述@Stuart Grassie的答案。

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

答案 2 :(得分:0)

通过Mock.Of<...>(...)的{​​{1}}方法,您可以使用async

Task.FromResult(...)