当前网站开发什么语言,php网站如何上传数据库,成都网站建设司,手机如何免费做网站前言相信大家在使用ASP.NET Core进行开发的时候#xff0c;肯定会涉及到读取Request.Body的场景#xff0c;毕竟我们大部分的POST请求都是将数据存放到Http的Body当中。因为笔者日常开发所使用的主要也是ASP.NET Core所以笔者也遇到这这种场景#xff0c;关于本篇文章所套路… 前言 相信大家在使用ASP.NET Core进行开发的时候肯定会涉及到读取Request.Body的场景毕竟我们大部分的POST请求都是将数据存放到Http的Body当中。因为笔者日常开发所使用的主要也是ASP.NET Core所以笔者也遇到这这种场景关于本篇文章所套路的内容来自于在开发过程中我遇到的关于Request.Body的读取问题。在之前的使用的时候基本上都是借助搜索引擎搜索的答案并没有太关注这个发现自己理解的和正确的使用之间存在很大的误区。故有感而发便写下此文以作记录。学无止境愿与君共勉。常用读取方式当我们要读取Request Body的时候相信大家第一直觉和笔者是一样的这有啥难的直接几行代码写完这里我们模拟在Filter中读取Request Body在Action或Middleware或其他地方读取类似有Request的地方就有Body如下所示public override void OnActionExecuting(ActionExecutingContext context)
{//在ASP.NET Core中Request Body是Stream的形式StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body stream.ReadToEnd();_logger.LogDebug(body content: body);base.OnActionExecuting(context);
}
写完之后也没多想毕竟这么常规的操作信心满满运行起来调试一把发现直接报一个这个错System.InvalidOperationException: Synchronous operations are disallowed. Call ReadAsync or set AllowSynchronousIO to true instead.大致的意思就是同步操作不被允许请使用ReadAsync的方式或设置AllowSynchronousIO为true。虽然没说怎么设置AllowSynchronousIO不过我们借助搜索引擎是我们最大的强项。同步读取首先我们来看设置AllowSynchronousIO为true的方式看名字也知道是允许同步IO设置方式大致有两种待会我们会通过源码来探究一下它们直接有何不同我们先来看一下如何设置AllowSynchronousIO的值。第一种方式是在ConfigureServices中配置操作如下services.ConfigureKestrelServerOptions(options
{options.AllowSynchronousIO true;
});
这种方式和在配置文件中配置Kestrel选项配置是一样的只是方式不同设置完之后即可运行不在报错。还有一种方式可以不用在ConfigureServices中设置通过IHttpBodyControlFeature的方式设置具体如下public override void OnActionExecuting(ActionExecutingContext context)
{var syncIOFeature context.HttpContext.Features.GetIHttpBodyControlFeature();if (syncIOFeature ! null){syncIOFeature.AllowSynchronousIO true;}StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body stream.ReadToEnd();_logger.LogDebug(body content: body);base.OnActionExecuting(context);
}
这种方式同样有效通过这种方式操作不需要每次读取Body的时候都去设置只要在准备读取Body之前设置一次即可。这两种方式都是去设置AllowSynchronousIO为true但是我们需要思考一点微软为何设置AllowSynchronousIO默认为false说明微软并不希望我们去同步读取Body。通过查找资料得出了这么一个结论Kestrel默认情况下禁用 AllowSynchronousIO同步IO线程不足会导致应用崩溃而同步I/O API例如HttpRequest.Body.Read是导致线程不足的常见原因。由此可以知道这种方式虽然能解决问题但是性能并不是不好微软也不建议这么操作当程序流量比较大的时候很容易导致程序不稳定甚至崩溃。异步读取通过上面我们了解到微软并不希望我们通过设置AllowSynchronousIO的方式去操作因为会影响性能。那我们可以使用异步的方式去读取这里所说的异步方式其实就是使用Stream自带的异步方法去读取如下所示public override void OnActionExecuting(ActionExecutingContext context)
{StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body stream.ReadToEndAsync().GetAwaiter().GetResult();_logger.LogDebug(body content: body);base.OnActionExecuting(context);
}
就这么简单不需要额外设置其他的东西仅仅通过ReadToEndAsync的异步方法去操作。ASP.NET Core中许多操作都是异步操作甚至是过滤器或中间件都可以直接返回Task类型的方法因此我们可以直接使用异步操作public override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
{StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body await stream.ReadToEndAsync();_logger.LogDebug(body content: body);await next();
}
这两种方式的操作优点是不需要额外设置别的只是通过异步方法读取即可也是我们比较推荐的做法。比较神奇的是我们只是将StreamReader的ReadToEnd替换成ReadToEndAsync方法就皆大欢喜了有没有感觉到比较神奇。当我们感到神奇的时候是因为我们对它还不够了解接下来我们就通过源码的方式一步一步的揭开它神秘的面纱。重复读取上面我们演示了使用同步方式和异步方式读取RequestBody但是这样真的就可以了吗其实并不行这种方式每次请求只能读取一次正确的Body结果如果继续对RequestBody这个Stream进行读取将读取不到任何内容首先来举个例子public override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
{StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body await stream.ReadToEndAsync();_logger.LogDebug(body content: body);StreamReader stream2 new StreamReader(context.HttpContext.Request.Body);string body2 await stream2.ReadToEndAsync();_logger.LogDebug(body2 content: body2);await next();
}
上面的例子中body里有正确的RequestBody的结果但是body2中是空字符串。这个情况是比较糟糕的为啥这么说呢如果你是在Middleware中读取的RequestBody而这个中间件的执行是在模型绑定之前那么将会导致模型绑定失败因为模型绑定有的时候也需要读取RequestBody获取http请求内容。至于为什么会这样相信大家也有了一定的了解因为我们在读取完Stream之后此时的Stream指针位置已经在Stream的结尾处即Position此时不为0而Stream读取正是依赖Position来标记外部读取Stream到啥位置所以我们再次读取的时候会从结尾开始读也就读取不到任何信息了。所以我们要想重复读取RequestBody那么就要再次读取之前重置RequestBody的Position为0如下所示public override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
{StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body await stream.ReadToEndAsync();_logger.LogDebug(body content: body);//或者使用重置Position的方式 context.HttpContext.Request.Body.Position 0;//如果你确定上次读取完之后已经重置了Position那么这一句可以省略context.HttpContext.Request.Body.Seek(0, SeekOrigin.Begin);StreamReader stream2 new StreamReader(context.HttpContext.Request.Body);string body2 await stream2.ReadToEndAsync();//用完了我们尽量也重置一下自己的坑自己填context.HttpContext.Request.Body.Seek(0, SeekOrigin.Begin);_logger.LogDebug(body2 content: body2);await next();
}
写完之后开开心心的运行起来看一下效果发现报了一个错System.NotSupportedException: Specified method is not supported.at Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.HttpRequestStream.Seek(Int64 offset, SeekOrigin origin)大致可以理解起来不支持这个操作至于为啥一会解析源码的时候咱们一起看一下。说了这么多那到底该如何解决呢也很简单微软知道自己刨下了坑自然给我们提供了解决办法用起来也很简单就是加EnableBufferingpublic override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
{//操作Request.Body之前加上EnableBuffering即可context.HttpContext.Request.EnableBuffering();StreamReader stream new StreamReader(context.HttpContext.Request.Body);string body await stream.ReadToEndAsync();_logger.LogDebug(body content: body);context.HttpContext.Request.Body.Seek(0, SeekOrigin.Begin);StreamReader stream2 new StreamReader(context.HttpContext.Request.Body);//注意这里我已经使用了同步读取的方式string body2 stream2.ReadToEnd();context.HttpContext.Request.Body.Seek(0, SeekOrigin.Begin);_logger.LogDebug(body2 content: body2);await next();
}
通过添加Request.EnableBuffering()我们就可以重复的读取RequestBody了看名字我们可以大概的猜出来他是和缓存RequestBody有关需要注意的是Request.EnableBuffering()要加在准备读取RequestBody之前才有效果否则将无效而且每次请求只需要添加一次即可。而且大家看到了我第二次读取Body的时候使用了同步的方式去读取的RequestBody是不是很神奇待会的时候我们会从源码的角度分析这个问题。源码探究上面我们看到了通过StreamReader的ReadToEnd同步读取Request.Body需要设置AllowSynchronousIO为true才能操作但是使用StreamReader的ReadToEndAsync方法却可以直接操作。StreamReader和Stream的关系我们看到了都是通过操作StreamReader的方法即可那关我Request.Body啥事别急咱们先看一看这里的操作首先来大致看下ReadToEnd的实现了解一下StreamReader到底和Stream有啥关联找到ReadToEnd方法[点击查看源码????[1]]public override string ReadToEnd()
{ThrowIfDisposed();CheckAsyncTaskInProgress();// 调用ReadBuffer然后从charBuffer中提取数据。StringBuilder sb new StringBuilder(_charLen - _charPos);do{//循环拼接读取内容sb.Append(_charBuffer, _charPos, _charLen - _charPos);_charPos _charLen; //读取buffer这是核心操作ReadBuffer();} while (_charLen 0);//返回读取内容return sb.ToString();
}
通过这段源码我们了解到了这么个信息一个是StreamReader的ReadToEnd其实本质是通过循环读取ReadBuffer然后通过StringBuilder去拼接读取的内容核心是读取ReadBuffer方法由于代码比较多我们找到大致呈现一下核心操作[点击查看源码????[2]]if (_checkPreamble)
{//通过这里我们可以知道本质就是使用要读取的Stream里的Read方法int len _stream.Read(_byteBuffer, _bytePos, _byteBuffer.Length - _bytePos);if (len 0){if (_byteLen 0){_charLen _decoder.GetChars(_byteBuffer, 0, _byteLen, _charBuffer, _charLen);_bytePos _byteLen 0;}return _charLen;}_byteLen len;
}
else
{//通过这里我们可以知道本质就是使用要读取的Stream里的Read方法_byteLen _stream.Read(_byteBuffer, 0, _byteBuffer.Length);if (_byteLen 0) {return _charLen;}
}
通过上面的代码我们可以了解到StreamReader其实是工具类只是封装了对Stream的原始操作简化我们的代码ReadToEnd方法本质是读取Stream的Read方法。接下来我们看一下ReadToEndAsync方法的具体实现[点击查看源码????[3]]public override Taskstring ReadToEndAsync()
{if (GetType() ! typeof(StreamReader)){return base.ReadToEndAsync();}ThrowIfDisposed();CheckAsyncTaskInProgress();//本质是ReadToEndAsyncInternal方法Taskstring task ReadToEndAsyncInternal();_asyncReadTask task;return task;
}private async Taskstring ReadToEndAsyncInternal()
{//也是循环拼接读取的内容StringBuilder sb new StringBuilder(_charLen - _charPos);do{int tmpCharPos _charPos;sb.Append(_charBuffer, tmpCharPos, _charLen - tmpCharPos);_charPos _charLen; //核心操作是ReadBufferAsync方法await ReadBufferAsync(CancellationToken.None).ConfigureAwait(false);} while (_charLen 0);return sb.ToString();
}
通过这个我们可以看到核心操作是ReadBufferAsync方法代码比较多我们同样看一下核心实现[点击查看源码????[4]]byte[] tmpByteBuffer _byteBuffer;
//Stream赋值给tmpStream
Stream tmpStream _stream;
if (_checkPreamble)
{int tmpBytePos _bytePos;//本质是调用Stream的ReadAsync方法int len await tmpStream.ReadAsync(new Memorybyte(tmpByteBuffer, tmpBytePos, tmpByteBuffer.Length - tmpBytePos), cancellationToken).ConfigureAwait(false);if (len 0){if (_byteLen 0){_charLen _decoder.GetChars(tmpByteBuffer, 0, _byteLen, _charBuffer, _charLen);_bytePos 0; _byteLen 0;}return _charLen;}_byteLen len;
}
else
{//本质是调用Stream的ReadAsync方法_byteLen await tmpStream.ReadAsync(new Memorybyte(tmpByteBuffer), cancellationToken).ConfigureAwait(false);if (_byteLen 0) {return _charLen;}
}
通过上面代码我可以了解到StreamReader的本质就是读取Stream的包装核心方法还是来自Stream本身。我们之所以大致介绍了StreamReader类就是为了给大家呈现出StreamReader和Stream的关系否则怕大家误解这波操作是StreamReader的里的实现而不是Request.Body的问题其实并不是这样的所有的一切都是指向Stream的Request的Body就是Stream这个大家可以自己查看一下了解到这一步我们就可以继续了。HttpRequest的Body上面我们说到了Request的Body本质就是StreamStream本身是抽象类所以Request.Body是Stream的实现类。默认情况下Request.Body的是HttpRequestStream的实例[点击查看源码????[5]]我们这里说了是默认因为它是可以改变的我们一会再说。我们从上面StreamReader的结论中得到ReadToEnd本质还是调用的Stream的Read方法即这里的HttpRequestStream的Read方法我们来看一下具体实现[点击查看源码????[6]]public override int Read(byte[] buffer, int offset, int count)
{//知道同步读取Body为啥报错了吧if (!_bodyControl.AllowSynchronousIO){throw new InvalidOperationException(CoreStrings.SynchronousReadsDisallowed);}//本质是调用ReadAsyncreturn ReadAsync(buffer, offset, count).GetAwaiter().GetResult();
}
通过这段代码我们就可以知道了为啥在不设置AllowSynchronousIO为true的情下读取Body会抛出异常了吧这个是程序级别的控制而且我们还了解到Read的本质还是在调用ReadAsync异步方法public override ValueTaskint ReadAsync(Memorybyte destination, CancellationToken cancellationToken default)
{return ReadAsyncWrapper(destination, cancellationToken);
}
ReadAsync本身并无特殊限制所以直接操作ReadAsync不会存在类似Read的异常。通过这个我们得出了结论Request.Body即HttpRequestStream的同步读取Read会抛出异常而异步读取ReadAsync并不会抛出异常只和HttpRequestStream的Read方法本身存在判断AllowSynchronousIO的值有关系。AllowSynchronousIO本质来源通过HttpRequestStream的Read方法我们可以知道AllowSynchronousIO控制了同步读取的方式。而且我们还了解到了AllowSynchronousIO有几种不同方式的去配置接下来我们来大致看下几种方式的本质是哪一种。通过HttpRequestStream我们知道Read方法中的AllowSynchronousIO的属性是来自IHttpBodyControlFeature也就是我们上面介绍的第二种配置方式private readonly HttpRequestPipeReader _pipeReader;
private readonly IHttpBodyControlFeature _bodyControl;
public HttpRequestStream(IHttpBodyControlFeature bodyControl, HttpRequestPipeReader pipeReader)
{_bodyControl bodyControl;_pipeReader pipeReader;
}
那么它和KestrelServerOptions肯定是有关系的因为我们只配置KestrelServerOptions的是HttpRequestStream的Read是不报异常的而HttpRequestStream的Read只依赖了IHttpBodyControlFeature的AllowSynchronousIO属性。Kestrel中HttpRequestStream初始化的地方在BodyControl[点击查看源码????[7]]private readonly HttpRequestStream _request;
public BodyControl(IHttpBodyControlFeature bodyControl, IHttpResponseControl responseControl)
{_request new HttpRequestStream(bodyControl, _requestReader);
}
而初始化BodyControl的地方在HttpProtocol中,我们找到初始化BodyControl的InitializeBodyControl方法[点击查看源码????[8]]public void InitializeBodyControl(MessageBody messageBody)
{if (_bodyControl null){//这里传递的是bodyControl传递的是this_bodyControl new BodyControl(bodyControl: this, this);}(RequestBody, ResponseBody, RequestBodyPipeReader, ResponseBodyPipeWriter) _bodyControl.Start(messageBody);_requestStreamInternal RequestBody;_responseStreamInternal ResponseBody;
}
这里我们可以看的到初始化IHttpBodyControlFeature既然传递的是this也就是HttpProtocol当前实例。也就是说HttpProtocol是实现了IHttpBodyControlFeature接口HttpProtocol本身是partial的我们在其中一个分布类HttpProtocol.FeatureCollection中看到了实现关系 [点击查看源码????[9]]internal partial class HttpProtocol : IHttpRequestFeature, IHttpRequestBodyDetectionFeature, IHttpResponseFeature, IHttpResponseBodyFeature, IRequestBodyPipeFeature, IHttpUpgradeFeature, IHttpConnectionFeature, IHttpRequestLifetimeFeature, IHttpRequestIdentifierFeature, IHttpRequestTrailersFeature, IHttpBodyControlFeature, IHttpMaxRequestBodySizeFeature, IEndpointFeature, IRouteValuesFeature { bool IHttpBodyControlFeature.AllowSynchronousIO { get AllowSynchronousIO; set AllowSynchronousIO value; } }
通过这个可以看出HttpProtocol确实实现了IHttpBodyControlFeature接口接下来我们找到初始化AllowSynchronousIO的地方找到了AllowSynchronousIO ServerOptions.AllowSynchronousIO;这段代码说明来自于ServerOptions这个属性找到初始化ServerOptions的地方[点击查看源码????[10]]private HttpConnectionContext _context;
//ServiceContext初始化来自HttpConnectionContext
public ServiceContext ServiceContext _context.ServiceContext;
protected KestrelServerOptions ServerOptions { get; set; } default!;
public void Initialize(HttpConnectionContext context)
{_context context;//来自ServiceContextServerOptions ServiceContext.ServerOptions;Reset();HttpResponseControl this;
}
通过这个我们知道ServerOptions来自于ServiceContext的ServerOptions属性我们找到给ServiceContext赋值的地方在KestrelServerImpl的CreateServiceContext方法里[点击查看源码????[11]]精简一下逻辑抽出来核心内容大致实现如下public KestrelServerImpl(IOptionsKestrelServerOptions options,IEnumerableIConnectionListenerFactory transportFactories,ILoggerFactory loggerFactory) //注入进来的IOptionsKestrelServerOptions调用了CreateServiceContext: this(transportFactories, null, CreateServiceContext(options, loggerFactory))
{
}private static ServiceContext CreateServiceContext(IOptionsKestrelServerOptions options, ILoggerFactory loggerFactory)
{//值来自于IOptionsKestrelServerOptions var serverOptions options.Value ?? new KestrelServerOptions();return new ServiceContext{Log trace,HttpParser new HttpParserHttp1ParsingHandler(trace.IsEnabled(LogLevel.Information)),Scheduler PipeScheduler.ThreadPool,SystemClock heartbeatManager,DateHeaderValueManager dateHeaderValueManager,ConnectionManager connectionManager,Heartbeat heartbeat,//赋值操作ServerOptions serverOptions,};
}
通过上面的代码我们可以看到如果配置了KestrelServerOptions那么ServiceContext的ServerOptions属性就来自于KestrelServerOptions即我们通过services.ConfigureKestrelServerOptions()配置的值总之得到了这么一个结论如果配置了KestrelServerOptions即services.Configure()那么AllowSynchronousIO来自于KestrelServerOptions。即IHttpBodyControlFeature的AllowSynchronousIO属性来自于KestrelServerOptions。如果没有配置那么直接通过修改IHttpBodyControlFeature实例的 AllowSynchronousIO属性能得到相同的效果毕竟HttpRequestStream是直接依赖的IHttpBodyControlFeature实例。EnableBuffering神奇的背后我们在上面的示例中看到了如果不添加EnableBuffering的话直接设置RequestBody的Position会报NotSupportedException这么一个错误而且加了它之后我居然可以直接使用同步的方式去读取RequestBody首先我们来看一下为啥会报错我们从上面的错误了解到错误来自于HttpRequestStream这个类[点击查看源码????[12]]上面我们也说了这个类继承了Stream抽象类,通过源码我们可以看到如下相关代码//不能使用Seek操作
public override bool CanSeek false;
//允许读
public override bool CanRead true;
//不允许写
public override bool CanWrite false;
//不能获取长度
public override long Length throw new NotSupportedException();
//不能读写Position
public override long Position
{get throw new NotSupportedException();set throw new NotSupportedException();
}
//不能使用Seek方法
public override long Seek(long offset, SeekOrigin origin)
{throw new NotSupportedException();
}
相信通过这些我们可以清楚的看到针对HttpRequestStream的设置或者写相关的操作是不被允许的这也是为啥我们上面直接通过Seek设置Position的时候为啥会报错还有一些其他操作的限制总之默认是不希望我们对HttpRequestStream做过多的操作特别是设置或者写相关的操作。但是我们使用EnableBuffering的时候却没有这些问题究竟是为什么?接下来我们要揭开它的什么面纱了。首先我们从Request.EnableBuffering()这个方法入手找到源码位置在HttpRequestRewindExtensions扩展类中[点击查看源码????[13]]我们从最简单的无参方法开始看到如下定义/// summary
/// 确保Request.Body可以被多次读取
/// /summary
/// param namerequest/param
public static void EnableBuffering(this HttpRequest request)
{BufferingHelper.EnableRewind(request);
}
上面的方法是最简单的形式还有一个EnableBuffering的扩展方法是参数最全的扩展方法这个方法可以控制读取的大小和控制是否存储到磁盘的限定大小/// summary
/// 确保Request.Body可以被多次读取
/// /summary
/// param namerequest/param
/// param namebufferThreshold内存中用于缓冲流的最大大小字节。较大的请求主体被写入磁盘。/param
/// param namebufferLimit请求正文的最大大小字节。尝试读取超过此限制将导致异常/param
public static void EnableBuffering(this HttpRequest request, int bufferThreshold, long bufferLimit)
{BufferingHelper.EnableRewind(request, bufferThreshold, bufferLimit);
}
无论那种形式最终都是在调用BufferingHelper.EnableRewind这个方法话不多说直接找到BufferingHelper这个类找到类的位置[点击查看源码????[14]]代码不多而且比较简洁咱们就把EnableRewind的实现粘贴出来//默认内存中可缓存的大小为30K,超过这个大小将会被存储到磁盘
internal const int DefaultBufferThreshold 1024 * 30;/// summary
/// 这个方法也是HttpRequest扩展方法
/// /summary
/// returns/returns
public static HttpRequest EnableRewind(this HttpRequest request, int bufferThreshold DefaultBufferThreshold, long? bufferLimit null)
{if (request null){throw new ArgumentNullException(nameof(request));}//先获取Request Bodyvar body request.Body;//默认情况Body是HttpRequestStream这个类CanSeek是false所以肯定会执行到if逻辑里面if (!body.CanSeek){//实例化了FileBufferingReadStream这个类看来这是关键所在var fileStream new FileBufferingReadStream(body, bufferThreshold,bufferLimit,AspNetCoreTempDirectory.TempDirectoryFactory);//赋值给Body也就是说开启了EnableBuffering之后Request.Body类型将会是FileBufferingReadStreamrequest.Body fileStream;//这里要把fileStream注册给Response便于释放request.HttpContext.Response.RegisterForDispose(fileStream);}return request;
}
从上面这段源码实现中我们可以大致得到两个结论•BufferingHelper的EnableRewind方法也是HttpRequest的扩展方法可以直接通过Request.EnableRewind的形式调用效果等同于调用Request.EnableBuffering因为EnableBuffering也是调用的EnableRewind•启用了EnableBuffering这个操作之后实际上会使用FileBufferingReadStream替换掉默认的HttpRequestStream所以后续处理RequestBody的操作将会是FileBufferingReadStream实例通过上面的分析我们也清楚的看到了核心操作在于FileBufferingReadStream这个类而且从名字也能看出来它肯定是也继承了Stream抽象类那还等啥直接找到FileBufferingReadStream的实现[点击查看源码????[15]]首先来看他类的定义public class FileBufferingReadStream : Stream
{
}
毋庸置疑确实是继承自Steam类我们上面也看到了使用了Request.EnableBuffering之后就可以设置和重复读取RequestBody说明进行了一些重写操作具体我们来看一下/// summary
/// 允许读
/// /summary
public override bool CanRead
{get { return true; }
}
/// summary
/// 允许Seek
/// /summary
public override bool CanSeek
{get { return true; }
}
/// summary
/// 不允许写
/// /summary
public override bool CanWrite
{get { return false; }
}
/// summary
/// 可以获取长度
/// /summary
public override long Length
{get { return _buffer.Length; }
}
/// summary
/// 可以读写Position
/// /summary
public override long Position
{get { return _buffer.Position; }set{ThrowIfDisposed();_buffer.Position value;}
}public override long Seek(long offset, SeekOrigin origin)
{//如果Body已释放则异常ThrowIfDisposed();//特殊情况抛出异常//_completelyBuffered代表是否完全缓存一定是在原始的HttpRequestStream读取完成后才置为true//出现没读取完成但是原始位置信息和当前位置信息不一致则直接抛出异常if (!_completelyBuffered origin SeekOrigin.End){throw new NotSupportedException(The content has not been fully buffered yet.);}else if (!_completelyBuffered origin SeekOrigin.Current offset Position Length){throw new NotSupportedException(The content has not been fully buffered yet.);}else if (!_completelyBuffered origin SeekOrigin.Begin offset Length){throw new NotSupportedException(The content has not been fully buffered yet.);}//充值buffer的Seekreturn _buffer.Seek(offset, origin);
}
因为重写了一些关键设置所以我们可以设置一些流相关的操作。从Seek方法中我们看到了两个比较重要的参数_completelyBuffered和_buffer_completelyBuffered用来判断原始的HttpRequestStream是否读取完成因为FileBufferingReadStream归根结底还是先读取了HttpRequestStream的内容。_buffer正是承载从HttpRequestStream读取的内容,我们大致抽离一下逻辑看一下切记这不是全部逻辑是抽离出来的大致思想private readonly ArrayPoolbyte _bytePool;
private const int _maxRentedBufferSize 1024 * 1024; //1MB
private Stream _buffer;
public FileBufferingReadStream(int memoryThreshold)
{//即使我们设置memoryThreshold那么它最大也不能超过1MB否则也会存储在磁盘上if (memoryThreshold _maxRentedBufferSize){_rentedBuffer bytePool.Rent(memoryThreshold);_buffer new MemoryStream(_rentedBuffer);_buffer.SetLength(0);}else{//超过1M将缓存到磁盘所以仅仅初始化_buffer new MemoryStream();}
}
这些都是一些初始化的操作核心操作当然还是在FileBufferingReadStream的Read方法里因为真正读取的地方就在这我们找到Read方法位置[点击查看源码????[16]]private readonly Stream _inner;
public FileBufferingReadStream(Stream inner)
{//接收原始的Request.Body_inner inner;
}
public override int Read(Spanbyte buffer)
{ThrowIfDisposed();//如果读取完成过则直接在buffer中获取信息直接返回if (_buffer.Position _buffer.Length || _completelyBuffered){return _buffer.Read(buffer);}//未读取完成才会走到这里//_inner正是接收的原始的RequestBody//读取的RequestBody放入buffer中var read _inner.Read(buffer);//超过设定的长度则会抛出异常if (_bufferLimit.HasValue _bufferLimit - read _buffer.Length){throw new IOException(Buffer limit exceeded.);}//如果设定存储在内存中并且Body长度大于设定的可存储在内存中的长度,则存储到磁盘中if (_inMemory _memoryThreshold - read _buffer.Length){_inMemory false;//缓存原始的Body流var oldBuffer _buffer;//创建缓存文件_buffer CreateTempFile();//超过内存存储限制但是还未写入过临时文件if (_rentedBuffer null){oldBuffer.Position 0;var rentedBuffer _bytePool.Rent(Math.Min((int)oldBuffer.Length, _maxRentedBufferSize));try{//将Body流读取到缓存文件流中var copyRead oldBuffer.Read(rentedBuffer);//判断是否读取到结尾while (copyRead 0){//将oldBuffer写入到缓存文件流_buffer当中_buffer.Write(rentedBuffer.AsSpan(0, copyRead));copyRead oldBuffer.Read(rentedBuffer);}}finally{//读取完成之后归还临时缓冲区到ArrayPool中_bytePool.Return(rentedBuffer);}}else{_buffer.Write(_rentedBuffer.AsSpan(0, (int)oldBuffer.Length));_bytePool.Return(_rentedBuffer);_rentedBuffer null;}}//如果读取RequestBody未到结尾,则一直写入到缓存区if (read 0){_buffer.Write(buffer.Slice(0, read));}else{//如果已经读取RequestBody完毕也就是写入到缓存完毕则更新_completelyBuffered//标记为以全部读取RequestBody完成后续在读取RequestBody则直接在_buffer中读取_completelyBuffered true;}//返回读取的byte个数用于外部StreamReader判断读取是否完成return read;
}
代码比较多看着也比较复杂其实核心思路还是比较清晰的我们来大致的总结一下•首先判断是否完全的读取过原始的RequestBody,如果完全完整的读取过RequestBody则直接在缓冲区中获取返回•如果RequestBody长度大于设定的内存存储限定则将缓冲写入磁盘临时文件中•如果是首次读取或为完全完整的读取完成RequestBody那么将RequestBody的内容写入到缓冲区知道读取完成其中CreateTempFile这是创建临时文件的操作流目的是为了将RequestBody的信息写入到临时文件中。可以指定临时文件的地址若如果不指定则使用系统默认目录它的实现如下[点击查看源码????[17]]private Stream CreateTempFile()
{//判断是否制定过缓存目录没有的话则使用系统临时文件目录if (_tempFileDirectory null){Debug.Assert(_tempFileDirectoryAccessor ! null);_tempFileDirectory _tempFileDirectoryAccessor();Debug.Assert(_tempFileDirectory ! null);}//临时文件的完整路径_tempFileName Path.Combine(_tempFileDirectory, ASPNETCORE_ Guid.NewGuid().ToString() .tmp);//返回临时文件的操作流return new FileStream(_tempFileName, FileMode.Create, FileAccess.ReadWrite, FileShare.Delete, 1024 * 16,FileOptions.Asynchronous | FileOptions.DeleteOnClose | FileOptions.SequentialScan);
}
我们上面分析了FileBufferingReadStream的Read方法这个方法是同步读取的方法可供StreamReader的ReadToEnd方法使用当然它还存在一个异步读取方法ReadAsync供StreamReader的ReadToEndAsync方法使用。这两个方法的实现逻辑是完全一致的只是读取和写入操作都是异步的操作这里咱们就不介绍那个方法了有兴趣的同学可以自行了解一下ReadAsync方法的实现[点击查看源码????[18]]当开启EnableBuffering的时候无论首次读取是设置了AllowSynchronousIO为true的ReadToEnd同步读取方式还是直接使用ReadToEndAsync的异步读取方式那么再次使用ReadToEnd同步方式去读取Request.Body也便无需去设置AllowSynchronousIO为true。因为默认的Request.Body已经由HttpRequestStream实例替换为FileBufferingReadStream实例而FileBufferingReadStream重写了Read和ReadAsync方法并不存在不允许同步读取的限制。总结 本篇文章篇幅比较多如果你想深入的研究相关逻辑希望本文能给你带来一些阅读源码的指导。为了防止大家深入文章当中而忘记了具体的流程逻辑在这里我们就大致的总结一下关于正确读取RequestBody的全部结论•首先关于同步读取Request.Body由于默认的RequestBody的实现是HttpRequestStream但是HttpRequestStream在重写Read方法的时候会判断是否开启AllowSynchronousIO如果未开启则直接抛出异常。但是HttpRequestStream的ReadAsync方法并无这种限制所以使用异步方式的读取RequestBody并无异常。•虽然通过设置AllowSynchronousIO或使用ReadAsync的方式我们可以读取RequestBody但是RequestBody无法重复读取这是因为HttpRequestStream的Position和Seek都是不允许进行修改操作的设置了会直接抛出异常。为了可以重复读取我们引入了Request的扩展方法EnableBuffering通过这个方法我们可以重置读取位置来实现RequestBody的重复读取。•关于开启EnableBuffering方法每次请求设置一次即可即在准备读取RequestBody之前设置。其本质其实是使用FileBufferingReadStream代替默认RequestBody的默认类型HttpRequestStream这样我们在一次Http请求中操作Body的时候其实是操作FileBufferingReadStream这个类重写Stream的时候Position和Seek都是可以设置的这样我们就实现了重复读取。•FileBufferingReadStream带给我们的不仅仅是可重复读取还增加了对RequestBody的缓存功能使得我们在一次请求中重复读取RequestBody的时候可以在Buffer里直接获取缓存内容而Buffer本身是一个MemoryStream。当然我们也可以自己实现一套逻辑来替换Body只要我们重写的时候让这个Stream支持重置读取位置即可。以上就是本次笔者对关于如何更好的方式操作Request.Body的理解关于讲解内容笔者深知自己能力有限理解的不一定透彻甚至理解的不一定对还望大家多多谅解也欢迎大家能够多多交流。References[1] 点击查看源码????: https://github.com/dotnet/runtime/blob/v5.0.5/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs#L393[2] 点击查看源码????: https://github.com/dotnet/runtime/blob/v5.0.5/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs#L576[3] 点击查看源码????: https://github.com/dotnet/runtime/blob/v5.0.5/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs#L898[4] 点击查看源码????: https://github.com/dotnet/runtime/blob/v5.0.5/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs#L1222[5] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpRequestStream.cs[6] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpRequestStream.cs#L55[7] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Infrastructure/BodyControl.cs[8] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpProtocol.cs#L315[9] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpProtocol.FeatureCollection.cs[10] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpProtocol.cs#L82[11] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/KestrelServerImpl.cs#L92[12] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Servers/Kestrel/Core/src/Internal/Http/HttpRequestStream.cs[13] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/Http/src/Extensions/HttpRequestRewindExtensions.cs[14] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/Http/src/Internal/BufferingHelper.cs[15] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/WebUtilities/src/FileBufferingReadStream.cs[16] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/WebUtilities/src/FileBufferingReadStream.cs#L212[17] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/WebUtilities/src/FileBufferingReadStream.cs#L198[18] 点击查看源码????: https://github.com/dotnet/aspnetcore/blob/v5.0.5/src/Http/WebUtilities/src/FileBufferingReadStream.cs#L285????欢迎扫码关注我的公众号????