我正在尝试在中间件中读取正文以进行身份验证,但是当请求到达api控制器时,对象为空,因为正在读取正文。有没有办法解决。我正在中间件中读这样的身体。
var buffer = new byte[ Convert.ToInt32( context.Request.ContentLength ) ];
await context.Request.Body.ReadAsync( buffer, 0, buffer.Length );
var body = Encoding.UTF8.GetString( buffer );
答案 0 :(得分:29)
如果您使用的是application/x-www-form-urlencoded
或multipart/form-data
,则可以安全地多次调用context.Request.ReadFormAsync()
,因为它会在后续调用中返回缓存的实例。
如果您使用的是其他内容类型,则必须手动缓冲请求,并使用MemoryStream
之类的可重绕流替换请求正文。以下是使用内联中间件的方法(您需要在管道中尽快注册):
app.Use(next => async context => {
// Keep the original stream in a separate
// variable to restore it later if necessary.
var stream = context.Request.Body;
// Optimization: don't buffer the request if
// there was no stream or if it is rewindable.
if (stream == Stream.Null || stream.CanSeek) {
await next(context);
return;
}
try {
using (var buffer = new MemoryStream()) {
// Copy the request stream to the memory stream.
await stream.CopyToAsync(buffer);
// Rewind the memory stream.
buffer.Position = 0L;
// Replace the request stream by the memory stream.
context.Request.Body = buffer;
// Invoke the rest of the pipeline.
await next(context);
}
}
finally {
// Restore the original stream.
context.Request.Body = stream;
}
});
您还可以使用BufferingHelper.EnableRewind()
扩展,它是Microsoft.AspNet.Http
包的一部分:它基于类似的方法,但依赖于一个特殊的流,它开始缓冲内存中的数据并将所有内容假脱机达到阈值时磁盘上的临时文件:
app.Use(next => context => {
context.Request.EnableRewind();
return next(context);
});
仅供参考:将来可能会将缓冲中间件添加到vNext。
答案 1 :(得分:7)
PinPoint提及EnableRewind
的用法Startup.cs
using Microsoft.AspNetCore.Http.Internal;
Startup.Configure(...){
...
//Its important the rewind us added before UseMvc
app.Use(next => context => { context.Request.EnableRewind(); return next(context); });
app.UseMvc()
...
}
然后在您的中间件中,您只需快退并重读
private async Task GenerateToken(HttpContext context)
{
context.Request.EnableRewind();
string jsonData = new StreamReader(context.Request.Body).ReadToEnd();
...
}
答案 2 :(得分:4)
这适用于.Net Core 2.1及更高版本。
今天我遇到了类似的问题。长话短说,曾经使用
Body.Seek(0, SeekOrigin.Begin);
今天例外,至少在我的情况下。将代码迁移到最新版本的.NET Core后,便会发生这种情况。
对我来说,解决方法是添加以下内容:
app.Use(next => context => { context.Request.EnableBuffering(); return next(context);
在设置控制器或MVC之前添加此项。这似乎是.NET Core 2.1版本的一部分。
希望这对某人有帮助!
欢呼和快乐的编码。