c# - 模拟 HttpMessageHandler 时未调用 SendAsync

标签 c# .net .net-core moq dotnet-httpclient

我正在尝试模拟 HttpMessageHandler 以便使用 HttpClient 测试类。

模拟似乎有效并且结果不错,但是当验证方法 SendAsunc 被调用的次数时,它返回 0 而不是 1...

知道为什么吗?我是 Moq 的新手,不了解它的所有复杂之处。

测试方法

[TestMethod]
public void LoginTest_ShouldTrue()
{
    // Setup the handler
    var mockHandler = new Mock<HttpMessageHandler>(MockBehavior.Strict);
    mockHandler.Protected()
        // Setup the protected method to mock
        .Setup<Task<HttpResponseMessage>>(
            "SendAsync",
            ItExpr.IsAny<HttpRequestMessage>(),
            ItExpr.IsAny<CancellationToken>()
        )
        // prepare the expected response of the mocked http call
        .ReturnsAsync(new HttpResponseMessage()
        {
            StatusCode = System.Net.HttpStatusCode.OK,
            Content = new StringContent("You are connected as xxxxxxxxxxxxxxxx.")
        })
        .Verifiable();

    // Use HttpClient with mocked HttpMessageHandler
    var httpClient = new HttpClient(mockHandler.Object);

    var objectToTest = new LoginApi(new Uri("https://test.com/"), ref httpClient);

    // Perform test
    var user = "test_user";
    var pass = "test_password_123";
    objectToTest.LoginAsync(user, pass).Result.Should().Be(true);
    objectToTest.IsLoggedIn.Should().Be(true);

    // Check the http call was as expected
    var expectedUri = new Uri("https://test.com/cgi/session.pl");
    var formVariables = new List<KeyValuePair<string, string>>(3);
    formVariables.Add(new KeyValuePair<string, string>(".submit", "Sign+in"));
    formVariables.Add(new KeyValuePair<string, string>("user_id", user));
    formVariables.Add(new KeyValuePair<string, string>("password", pass));
    var expectedContent = new FormUrlEncodedContent(formVariables);
    mockHandler.Protected().Verify<Task<HttpResponseMessage>>(
        "SendAsync",
        Times.Exactly(1), // We expected a single external request
        ItExpr.Is<HttpRequestMessage>(req =>
            req.Method == HttpMethod.Post // We expected a POST request
            && req.RequestUri == expectedUri // to this uri
            && req.Content == expectedContent // with this content
        ),
        ItExpr.IsAny<CancellationToken>()
    );
}

测试的方法

public class LoginApi : ILoginApi
{
    private readonly Uri baseUri;
    private readonly HttpClient client;

    public bool IsLoggedIn { get; protected set; }

    public LoginApi(Uri baseUri, ref HttpClient client)
    {
        this.baseUri = baseUri;
        this.client = client;
        IsLoggedIn = false;
    }

    public async Task<bool> LoginAsync(string user, string pass)
    {
        var formVariables = new List<KeyValuePair<string, string>>(3);
        formVariables.Add(new KeyValuePair<string, string>(".submit", "Sign+in"));
        formVariables.Add(new KeyValuePair<string, string>("user_id", user));
        formVariables.Add(new KeyValuePair<string, string>("password", pass));

        var targetUri = new Uri(baseUri, string.Join('/', URLs.ServiceCgiSuffix, "session.pl"));
        var res = await client.PostAsync(targetUri, new FormUrlEncodedContent(formVariables));

        var content = await res.Content.ReadAsStringAsync();
        IsLoggedIn = content.Contains("You are connected as");
        return IsLoggedIn;
    }

    public async Task<bool> LogoutAsync()
    {
        var formVariables = new List<KeyValuePair<string, string>>(1);
        formVariables.Add(new KeyValuePair<string, string>(".submit", "Sign-out"));

        var targetUri = new Uri(baseUri, string.Join('/', URLs.ServiceCgiSuffix, "session.pl"));
        var res = await client.PostAsync(targetUri, new FormUrlEncodedContent(formVariables));

        var content = await res.Content.ReadAsStringAsync();
        IsLoggedIn = !content.Contains("See you soon!");
        return !IsLoggedIn;
    }
}

错误详情

StackTrace: 
at Moq.Mock.VerifyCalls(Mock targetMock, InvocationShape expectation, LambdaExpression expression, Times times, String failMessage) in C:\projects\moq4\src\Moq\Mock.cs:line 414
    at Moq.Mock.Verify[T,TResult](Mock`1 mock, Expression`1 expression, Times times, String failMessage) in C:\projects\moq4\src\Moq\Mock.cs:line 315
    at Moq.Protected.ProtectedMock`1.Verify[TResult](String methodName, Times times, Object[] args) in C:\projects\moq4\src\Moq\Protected\ProtectedMock.cs:line 171
    at OpenFoodFacts.Test.Login.LoginApiTest.LoginTest_ShouldTrue() in path\to\project\LoginApiTest.cs:line 56
Result message: 
Test method OpenFoodFacts.Test.Login.LoginApiTest.LoginTest_ShouldTrue threw exception: 
Moq.MockException: 
Expected invocation on the mock exactly 1 times, but was 0 times:
mock => mock.SendAsync(
    It.Is<HttpRequestMessage>(req => (req.Method == POST && req.RequestUri == https://test.com/cgi/session.pl) && req.Content == FormUrlEncodedContent),
    It.IsAny<CancellationToken>())

Configured setups: 
HttpMessageHandler mock => mock.SendAsync(It.IsAny<HttpRequestMessage>(), It.IsAny<CancellationToken>())

Performed invocations: 
HttpMessageHandler.SendAsync(Method: POST, RequestUri: 'https://test.com/cgi/session.pl', Version: 2.0, Content: System.Net.Http.FormUrlEncodedContent, Headers:
{
    Content-Type: application/x-www-form-urlencoded
}, CancellationToken)

最佳答案

问题在于请求内容是否相等,如果两个操作数引用同一个对象,则使用==运算符比较两个对象返回true;他们没有。如果你删除了这一行 “&& req.Content == expectedContent” 你的测试会工作正常。除此之外,您的测试和模拟设置运行良好。

关于c# - 模拟 HttpMessageHandler 时未调用 SendAsync,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52705482/

相关文章:

c# - 将工作单元与服务或 repo 分离

c# - winscard.dll 预热

c# - page.Isvalid 总是返回false?

c# - 为什么我的 C# 默认接口(interface)实现在具体类定义中没有被识别?

visual-studio-2017 - VS2017 nuget加载项目依赖失败

c# - 对 GridView 行执行操作的最佳方式

c# - 理解/可视化 C# 代码的工具

Java 是开源的,那又怎样?

.net - 开发人员是否必须关心将来更改 ORM 的可能性?

linux - 在 Linux 容器上的 dotnet Core 中实现消息消费者