1. ホーム
  2. c#

[解決済み】Moqを使用してユニットテストの非同期メソッドをモック化する

2022-04-07 04:15:44

質問

ウェブページを作成するサービスのメソッドをテストしています。 API を呼び出します。通常の HttpClient は、ウェブサービス (ソリューション内の別のプロジェクトにあります) をローカルで実行しても、ユニットテストでは問題なく動作します。

しかし、変更をチェックインすると、ビルドサーバーはウェブサービスにアクセスできないので、テストは失敗します。

この問題を回避するために、ユニットテストに IHttpClient インターフェイスを実装し、私のアプリケーションで使用するバージョンを実装しています。ユニットテストでは、非同期ポストメソッドをモック化したバージョンを作成します。ここで、問題が発生しました。私はOKを返したいのです。 HttpStatusResult この特定のテストのために。別の同じようなテストでは、悪い結果を返します。

テストは実行されますが、完了することはありません。awaitのところでハングアップしてしまいます。私は非同期プログラミング、デリゲート、そして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
}

これが私のユニットテストのメソッドです。

[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.");
}

何が間違っているのでしょうか?

よろしくお願いします。

解決方法は?

タスクを作成しても開始しないので、完了しない。しかし、ただタスクを開始するだけでなく、代わりに Task.FromResult<TResult> これは、すでに完了したタスクを表示します。

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

この方法では実際の非同期をテストしないことに注意してください。そうしたい場合は、もう少し手間をかけて Task<T> しかし、それはまた別の機会に。

のフェイクを使用することも検討した方がいいかもしれません。 IHttpClient は、すべてをモック化するのではなく、必要な頻度に依存します。