小编典典

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

all

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

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

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

测试将运行但永远不会完成。它挂在等待。我是异步编程、委托和 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.");
}

我究竟做错了什么?

谢谢您的帮助。


阅读 76

收藏
2022-06-04

共1个答案

小编典典

您正在创建任务但从未启动它,因此它永远不会完成。但是,不要只是开始任务 - 相反,更改为 using
Task.FromResult<TResult>which 会给你一个已经完成的任务:

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

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

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

2022-06-04