首页 > 解决方案 > 如何在单元测试中验证 Flurl Http 中的请求正文内容?

问题描述

我正在使用 Flurl Http 发出 http 请求。在单元测试中,我试图验证预期的内容是否已传递给发件人。我正在尝试这样:

httpTest.ShouldHaveCalled(url)
        .WithVerb(HttpMethod.Post)
        .WithContentType(contentType)
        .With(w => w.Request.Content.ReadAsStringAsync().Result == content)
        .Times(1);

但是,这失败了System.ObjectDisposedException Cannot access a disposed object. Object name: 'System.Net.Http.StringContent'.

看起来 Flurl 在测试中完成验证之前正在处理请求正文内容。如何捕获请求正文以进行验证?

编辑(一个完全可重现的例子):

using System;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;

using Autofac.Extras.Moq;

using Flurl.Http;
using Flurl.Http.Testing;

using Xunit;

namespace XUnitTestProject1
{
    public class MyClassTest : IDisposable
    {
        private readonly AutoMock container;
        private readonly HttpTest client;

        public MyClassTest()
        {
            this.container = AutoMock.GetLoose();
            this.client = new HttpTest();
        }

        [Fact]
        public async Task SendAsync_ValidateRequestBody()
        {
            const string url = "http://www.example.com";
            const string content = "Hello, world";

            var sut = this.container.Create<MyClass>();
            await sut.SendAsync(url, content);

            this.client.ShouldHaveCalled(url)
                .With(w => w.Request.Content.ReadAsStringAsync().Result == content);
        }

        public void Dispose()
        {
            this.container?.Dispose();
            this.client?.Dispose();
        }
    }

    public class MyClass
    {
        public virtual async Task SendAsync(string url, string content)
        {
            await url.PostAsync(new StringContent(content, Encoding.UTF8, "text/plain"));
        }
    }
}

标签: c#unit-testingflurl

解决方案


在大多数情况下(参见下面的编辑),Flurl 已经捕获了它,您只需要以不同的方式访问它。

在您的示例中,w.Request是 Flurl 公开的堆栈中的“原始” HttpRequestMessageHttpClient因此您可以在需要时进入引擎盖。HttpRequestMessage.Content是一个只读的、只进的流,在您访问它时已经被读取和释放。

要断言捕获的字符串主体,您通常只需这样做:

httpTest.ShouldHaveCalled(url)
    ...
    .WithRequestBody(content)

编辑

正如您所指出的,这不适用于您使用 Flurl 的方式。包含的字符串StringContent实际上是只写的,即没有属性将其公开以供读取。这就是 Flurl 的目的CapturedStringContent。如果您使用该类型作为 , 的直接替代品StringContentRequestBody将在您的测试中可用。

文档中没有很好地涵盖这一点的原因是,如果您以“Flurl 方式”做事,那么您一开始就没有明确地创建内容对象。PostStringAsync并且PostJsonAsync是发送 POST 请求的更常见的方法,两者都是使用CapturedStringContent. 如果可以,请使用其中一种方法,或者PostAsync(new CapturedStringContent(...))如果出于某种原因需要获取较低级别的内容对象,请使用这些方法。


推荐阅读