Skip to content

dotnet 默认创建的 JsonContent 没有 Content Length 的内容头

Updated: at 02:05,Created: at 23:07

本文记录一个 dotnet 的设计问题,默认创建出来的 JsonContent 对象的 Headers 里,是没有 Content-Length 信息的

如下面代码创建一个 JsonContent 对象

using System.Net.Http.Json;
var foo = new Foo();
var jsonContent = JsonContent.Create(foo);
class Foo
{
public int Value { set; get; }
}

此时如果遍历 JsonContent 的 Headers 属性,将只可以拿到 Content-Type 信息,没有 Content-Length 信息

在现代的绝大部分服务端,都是支持 Content 不带 Content-Length 信息的,这在大部分后台上都能正常符合预期工作

即使用大概如下代码的 JsonContent 发送出去的请求,在请求里面也是不带 Content-Length 信息的

var foo = new Foo();
var jsonContent = JsonContent.Create(foo);
var httpClient = new HttpClient();
await httpClient.PostAsync("https://blog.lindexi.com", jsonContent);

那如何可以让 JsonContent 带上 Content-Length 信息?只需调用 LoadIntoBufferAsync 方法,如以下代码

var jsonContent = JsonContent.Create(foo);
await jsonContent.LoadIntoBufferAsync();

调用完成 LoadIntoBufferAsync 方法,即可在 Headers 里面看到 Content-Length 信息,且使用如下代码发送请求也是带上 Content-Length 信息的

await jsonContent.LoadIntoBufferAsync();
var httpClient = new HttpClient();
await httpClient.PostAsync("https://blog.lindexi.com", jsonContent);

我查看请求的信息是通过自己创建一个简单的 ASP.NET Core 程序,代码大概如下

var builder = WebApplication.CreateSlimBuilder(args);
var app = builder.Build();
app.MapPost("/", async context =>
{
await Task.CompletedTask;
var headers = context.Request.Headers;
});
app.Run();

通过断点在 var headers = context.Request.Headers; 即可了解客户端请求发送过来的请求头信息

以及将此请求尝试发送到其他服务器上,通过抓包确定了具体的行为

这在 dotnet 里面认为设计如此,且认为如果没有足够多的报告说缺少 Content-Length 信息会让后台不工作,则依然保持此行为

讨论内容请看:

.NET 6: JsonContent.Create(obj) should set Content-Length HTTP request header · Issue #70793 · dotnet/runtime

Content-Length not appended when using JsonContent · Issue #82984 · dotnet/runtime

Provide better json support for servers don’t support chunked request body (re-open) · Issue #55583 · dotnet/runtime

本文代码放在 githubgitee 上,可以使用如下命令行拉取代码。我整个代码仓库比较庞大,使用以下命令行可以进行部分拉取,拉取速度比较快

先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin 1b312eb1bfb867e56c5bbc61df720819fe1e15fc

以上使用的是国内的 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码,将 gitee 源换成 github 源进行拉取代码。如果依然拉取不到代码,可以发邮件向我要代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin 1b312eb1bfb867e56c5bbc61df720819fe1e15fc

获取代码之后,进入 Workbench/CaiballkaylecaWairlaroweneno 文件夹,即可获取到源代码

更多技术博客,请参阅 博客导航


知识共享许可协议

原文链接: http://blog.lindexi.com/post/dotnet-%E9%BB%98%E8%AE%A4%E5%88%9B%E5%BB%BA%E7%9A%84-JsonContent-%E6%B2%A1%E6%9C%89-Content-Length-%E7%9A%84%E5%86%85%E5%AE%B9%E5%A4%B4

本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。 欢迎转载、使用、重新发布,但务必保留文章署名 林德熙 (包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我 联系