關於.Net Core下因客戶端主動取消導致Request請求[FromBody]模型綁定讀取異常BadHttpRequestException

来源:https://www.cnblogs.com/lxhbky/archive/2022/03/21/16033691.html
-Advertisement-
Play Games

Net6 Configuration & Options 源碼分析 Part2 Options 第二部分主要記錄Options 模型 OptionsConfigurationServiceCollectionExtensions類提供了對Options 模型與配置系統的Configure方法的擴展 ...


  最近我们的接口老是会出现BadHttpRequestException异常,但是手动查看报错的页面却一点问题没有,很奇怪,后来仔细研究这个异常。异常原因其实根据异常对象本身就已经能分析出来了(由于第一次遇到不清楚怎么造成的,尴尬),就是由于无法正常读取Http-Request请求对象里的内容导致的异常!但是什么情况下会导致这个情况,不得而知。  后来百度其他朋友的文章,终于了解到发生当前错误的原因:

  是由于当请求出现并发,而请求的线程池又不够使用,前面的请求阻塞后面的请求时,后面的请求强制取消后就会导致后面的请求在被接收程序处理Request内容时发生异常,即:BadHttpRequestException

先看下接口是怎么接收的:

[HttpPost]
public async Task<IActionResult> PPIFrameInit([FromBody] PaypalClientRequest parames)
{
      //...  
}

很显然,接口采用的是自动绑定Model模型Request-Body的方式,那么如果按照上述请求就很容易发生下面的异常了。

异常内容如下:

ClientIP:157.185.158.160 
URL:http://www.coowigsby.com/ajax/paydd/PPIFrameInit 
Type:Microsoft.AspNetCore.Server.Kestrel.Core.BadHttpRequestException 
Msg:Microsoft.AspNetCore.Server.Kestrel.Core.BadHttpRequestException: Unexpected end of request content.
   at Microsoft.AspNetCore.Server.Kestrel.Core.KestrelBadHttpRequestException.Throw(RequestRejectionReason reason)
   at Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.Http1ContentLengthMessageBody.ReadAsyncInternal(CancellationToken cancellationToken)
   at Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.HttpRequestStream.ReadAsyncInternal(Memory`1 destination, CancellationToken cancellationToken)
   at Microsoft.AspNetCore.WebUtilities.FileBufferingReadStream.ReadAsync(Memory`1 buffer, CancellationToken cancellationToken)
   at Microsoft.AspNetCore.WebUtilities.StreamHelperExtensions.DrainAsync(Stream stream, ArrayPool`1 bytePool, Nullable`1 limit, CancellationToken cancellationToken)
   at Microsoft.AspNetCore.Mvc.Formatters.NewtonsoftJsonInputFormatter.ReadRequestBodyAsync(InputFormatterContext context, Encoding encoding)
   at Microsoft.AspNetCore.Mvc.ModelBinding.Binders.BodyModelBinder.BindModelAsync(ModelBindingContext bindingContext)
   at Microsoft.AspNetCore.Mvc.ModelBinding.ParameterBinder.BindModelAsync(ActionContext actionContext, IModelBinder modelBinder, IValueProvider valueProvider, ParameterDescriptor parameter, ModelMetadata metadata, Object value, Object container)
   at Microsoft.AspNetCore.Mvc.Controllers.ControllerBinderDelegateProvider.<>c__DisplayClass0_0.<<CreateBinderDelegate>g__Bind|0>d.MoveNext()
--- End of stack trace from previous location ---
   at Microsoft.AspNetCore.Mvc.Infrastructure.ControllerActionInvoker.<InvokeInnerFilterAsync>g__Awaited|13_0(ControllerActionInvoker invoker, Task lastTask, State next, Scope scope, Object state, Boolean isCompleted)
   at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.<InvokeNextResourceFilter>g__Awaited|24_0(ResourceInvoker invoker, Task lastTask, State next, Scope scope, Object state, Boolean isCompleted)
   at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.Rethrow(ResourceExecutedContextSealed context)
   at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.Next(State& next, Scope& scope, Object& state, Boolean& isCompleted)
   at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.InvokeFilterPipelineAsync()
--- End of stack trace from previous location ---
   at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.<InvokeAsync>g__Awaited|17_0(ResourceInvoker invoker, Task task, IDisposable scope)
   at Microsoft.AspNetCore.Routing.EndpointMiddleware.<Invoke>g__AwaitRequestTask|6_0(Endpoint endpoint, Task requestTask, ILogger logger)
   at MeShop.CacheResponse.CacheResponseMiddleware.Invoke(HttpContext httpContext) in /builds/sitegroup/meshop/meshop/MeShop.CacheResponse/CacheResponseMiddleware.cs:line 56
   at MeShop.View.Shop.Models.CheckExpirationMiddleware.Invoke(HttpContext httpContext) in /builds/sitegroup/meshop/meshop/MeShop.View.Shop/Models/CheckExpirationMiddleware.cs:line 38
   at MeShop.WebCommon.Cache.ResponseDistributedCacheMiddleware.Invoke(HttpContext httpContext) in /builds/sitegroup/meshop/meshop/MeShop.WebCommon/Cache/ResponseDistributedCacheMiddleware.cs:line 105
   at Microsoft.AspNetCore.ResponseCaching.ResponseCachingMiddleware.Invoke(HttpContext httpContext)
   at Microsoft.AspNetCore.Authorization.AuthorizationMiddleware.Invoke(HttpContext context)
   at Microsoft.AspNetCore.Authentication.AuthenticationMiddleware.Invoke(HttpContext context)
   at MeShop.WebCommon.CircuitBreakerWebMiddleware.<>c__DisplayClass13_0.<<InvokeAsync>b__1>d.MoveNext() in /builds/sitegroup/meshop/meshop/MeShop.WebCommon/Middleware/CircuitBreakerWebMiddleware.cs:line 246
--- End of stack trace from previous location ---
   at Polly.AsyncPolicy.<>c__DisplayClass40_0.<<ImplementationAsync>b__0>d.MoveNext()
--- End of stack trace from previous location ---
   at Polly.CircuitBreaker.AsyncCircuitBreakerPolicy.<>c__DisplayClass8_0`1.<<ImplementationAsync>b__0>d.MoveNext()
--- End of stack trace from previous location ---
   at Polly.CircuitBreaker.AsyncCircuitBreakerEngine.ImplementationAsync[TResult](Func`3 action, Context context, CancellationToken cancellationToken, Boolean continueOnCapturedContext, ExceptionPredicates shouldHandleExceptionPredicates, ResultPredicates`1 shouldHandleResultPredicates, ICircuitController`1 breakerController)
   at Polly.CircuitBreaker.AsyncCircuitBreakerPolicy.ImplementationAsync[TResult](Func`3 action, Context context, CancellationToken cancellationToken, Boolean continueOnCapturedContext)
   at Polly.AsyncPolicy.ExecuteAsync(Func`3 action, Context context, CancellationToken cancellationToken, Boolean continueOnCapturedContext)
   at MeShop.WebCommon.CircuitBreakerWebMiddleware.InvokeAsync(HttpContext context) in /builds/sitegroup/meshop/meshop/MeShop.WebCommon/Middleware/CircuitBreakerWebMiddleware.cs:line 275

 

这里列举一下本地调试的复现步骤(本地调试是单线程):

  • 在请求处理过程的某处设置断点
  • 在PostMan或浏览器上先发送A请求,命中断点,不继续往下处理,让请求阻塞
  • 继续发放B请求,这时请求阻塞中,然后点击取消请求
  • 释放断点继续处理,结果是A请求正常处理结束,B请求处理时抛出异常

解决方法有三种(参考的博客里只介绍了一种):

  • 1_可选择在全局异常过滤器中不处理该异常,不打印错误日志(有掩耳盗铃的嫌疑,不推荐);
  • 2_可以将接口[FromBody]由自动绑定改成代码读取接收,使用try-catch捕获异常单独处理(跟1类似,也不推荐);
  • 3_可以增加最小线程池数量,使其能够拥有足够的线程处理并发请求(确实访问量太大,可以使用);
  • 4_优化现有的代码:分析当时的请求为何造成了线程不够用的原因,把耗时较长造成阻塞的有问题的方法处理掉,更新后再继续观察(适用于访问量没有那么大,部分接口代码质量不好的情况时使用);

 

参考博客:关于.Net Core3.0下因客户端主动取消请求导致的Request.Body异常

*感谢您的阅读。喜欢的、有用的就请大哥大嫂们高抬贵手“推荐一下”吧!你的精神 支持是博主强大的写作动力。欢迎转载!
*博主的文章是自己平时开发总结的经验,由于博主的水平不高,不足和错误之处在所难免,希望大家能够批评指出。
*我的博客: http://www.cnblogs.com/lxhbky/
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • hello大家好呀,我是小樓~ 今天又帶來一次性能優化的分享,這是我剛進公司時接手的祖傳(壞笑)項目,這個項目在我的文章中屢次被提及,我在它上面做了很多的性能優化,比如《記一次提升18倍的性能優化》這篇文章,比較偏向某個細節的優化,本文更偏向巨集觀上的性能優化,可以說是個老演員了。 背景 為了新朋友能 ...
  • 一、概述 date代表一個特定的時間,精確到毫秒 二、構造方法 方法名說明 public Date() 分配一個Date對象,並初始化,以便它代表被分配的時間,精確到毫秒 public Date(long date) 分配一個Date對象,並將其初始化為表示從標準基準時間起指定的毫秒數 import ...
  • .net core IdentityServer4 認證授權 token ...
  • 1.什麼是Blazor? 有什麼優勢? ASP.NET Core Blazor 簡介 Blazor 是一個使用 Blazor 生成互動式客戶端 Web UI 的框架: 使用 C# 代替 JavaScript 來創建信息豐富的互動式 UI。 共用使用 .NET 編寫的伺服器端和客戶端應用邏輯。 將 U ...
  • 配置源的同步 IOptionsMonitor 使用 //以下demo演示使用IOptionsMonitor重新載入配置並當重新載入配置是執行回調函數 var configuration = new ConfigurationBuilder().AddJsonFile(path: "profile.j ...
  • 十天前,我發佈了對.NET Core程式進行瘦身的開源軟體Zack.DotNetTrimmer,與.NET Core內置的剪裁器相比,Zack.DotNetTrimmer不僅對程式的剪裁效果更好,而且還支持WPF、WinForm程式。 很多朋友對於這個開源項目的原理很感興趣,因此我將通過這篇文章為大 ...
  • 1. 前言 通過之前的學習,我們已經瞭解了各參數以及配置的意義,接下來的文章我們分別從bombardier以及wrk入手,進一步瞭解彼此之間的聯繫 2. 認識 bombardier bombardier 是一個 HTTP(S) 基準測試工具。它是用 Go 編程語言編寫的,並使用優秀的fasthttp ...
  • 一 發佈者和訂閱者 很多時候都有這種需求,當一個特定的程式事件發生時,程式的其他部分可以得到該事件已經發生的通知。 發佈者/訂閱者模式可以滿足這種需求。 發佈者:發佈某個事件的類或結構,其他類可以在該事件發生時得到通知。 訂閱者:註冊併在事件發生時得到通知的類或結構。 事件處理程式:由訂閱者註冊到事 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...