ASP.NET Core應用錯誤處理之三種呈現錯誤頁面的方式

這篇文章主要給大家介紹了關於ASP.NET Core應用錯誤處理之三種呈現錯誤頁面的方式的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨着小編來一起學習學習吧

前言

由於ASP.NET Core應用是一個同時處理多個請求的服務器應用,所以在處理某個請求過程中拋出的異常並不會導致整個應用的終止。出於安全方面的考量,爲了避免敏感信息的外泄,客戶端在默認的情況下並不會得到詳細的出錯信息,這無疑會在開發環境下增加查錯糾錯的難度。對於生產環境來說,我們也希望最終用戶能夠根據具體的錯誤類型得到具有針對性並且友好的錯誤消息。ASP.NET Core提供了相應的中間件幫助我們將定製化的錯誤信息呈現出來,這些中間件都定義在“Microsoft.AspNetCore.Diagnostics”這個NuGet包中。在着重介紹這些中間件之前,我們照理演示幾個簡單的實例讓讀者朋友們對這些中間件的作用有一個大概的瞭解。

一、顯示開發者異常頁面

一般情況下,如果ASP.NET Core在處理某個請求時出現異常,它一般會返回一個狀態碼爲“500 Internal Server Error”的響應。爲了避免一些敏感信息的外泄,詳細的錯誤信息並不會隨着響應發送給客戶端,所以客戶端只會得到一個很一般化的錯誤消息。以如下這個程序爲例,服務端在處理每個請求時都會拋出一個類型爲InvalidOperationException的異常。

public class Program
 {
 public static void Main()
 {
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app.Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
 .Build()
 .Run();
 }
 }

當我們利用瀏覽器訪問這個應用的時候,總是會得到如下圖所示的這個錯誤頁面。可以看出這個頁面僅僅告訴我們目標應用當前無法正常處理本次請求,除了提供的響應狀態碼(“HTTP ERROR 500”)之外,它並沒有提供任何有益於差錯糾錯的錯誤信息。

那麼有人可能會覺得雖然瀏覽器上沒有顯示出任何詳細的錯誤信息,也許它會隱藏在接收到的HTTP響應報文中。針對通過瀏覽器放出的這個請求,得到的響應內容如下所示,我們會發現響應報文根本沒有主體部分,有限的幾個報頭也並沒有承載任何與錯誤有關的信息。

 HTTP/1.1 500 Internal Server Error
 Date: Fri, 09 Dec 2016 23:42:18 GMT
 Content-Length: 0
 Server: Kestrel

由於應用並沒有中斷,瀏覽器上也並沒有顯示任何具有針對性的錯誤信息,開發人員在進行查錯糾錯的時候如何準確定位到作爲錯誤根源的那一行代碼呢?具體來說,我們又兩種解決方案,一種就是利用日誌,因爲ASP.NET Core在進行請求處理時出現的任何錯誤都會被寫入日誌,所以我們可以通過註冊相應的LoggerProvider(比如註冊一個ConsoleLoggerProvider將日誌直接寫入宿主應用的控制檯)到來獲取寫入的錯誤日誌。

至於另一種解決方案,就是直接顯示一個包含錯誤相應信息的錯誤頁面,由於這個頁面是在開發環境給開發者看的,所以我們將這個頁面稱爲“開發者異常頁面(Developer Exception Page)”。針對頁面的自動呈現是利用一個名爲DeveloperExceptionPageMiddleware的中間件來完成的,我們可以調用ApplicationBuilder的擴展方法UseDeveloperExceptionPage來註冊這個中間件。

 public class Program
 {
 public static void Main()
 {
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app
 .UseDeveloperExceptionPage()
 .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
 .Build()
 .Run();
 }
 }

一旦註冊了這個DeveloperExceptionPageMiddleware中間件,ASP.NET Core應用在處理請求出現的異常信息就會以下圖的形式直接出現在瀏覽器上,我們可以在這個頁面中看到幾乎所有的錯誤信息,包括異常的類型、消息和堆棧信息等。

開發者異常頁面除了顯示與拋出的異常相關的信息之外,還會以如下圖所示的形式顯示與當前請求上下文相關的信息,其中包括當前請求URL攜帶的所有查詢字符串、所有請求報頭以及Cookie的內容。如此詳盡的信息無疑會極大地幫助開發人員儘快地找出錯誤的根源。

通過DeveloperExceptionPageMiddleware中間件呈現的錯誤頁面僅僅是供開發人員使用的,詳細的錯誤信息往往會攜帶一些敏感的信息,所以務必記住只有在開發環境才能註冊這個中間件,如下所示的代碼片段體現了針對DeveloperExceptionPageMiddleware中間件正確的註冊方式。

 new WebHostBuilder()
 .UseStartup<Startup>()
 …
 
 public class Startup
 {
 public void Configure(IApplicationBuilder app, IHostingEnvironment env)
 {
 if (env.IsDevelopment())
 {
 app.UseDeveloperExceptionPage();
 }
 }
 }

二、顯示定製異常頁面

DeveloperExceptionPageMiddleware中間件通過將異常詳細信息和基於當前請求的內容直接呈現在錯誤頁面中,這爲開發人員的糾錯診斷提供了極大的便利。但是在生產環境下,我們傾向於爲最終的用戶呈現一個定製的錯誤頁面,而這可以通過註冊另一個名爲ExceptionHandlerMiddleware的中間件來實現。顧名思義,這個中間件旨在提供一個異常處理器(Exception Handler)來處理拋出的異常。實際上這個所謂的異常處理器就是一個類型爲RequestDelegate的委託對象,ExceptionHandlerMiddleware中間件捕捉到拋出的異常後利用它來響應當前的請求。

還是以上面創建的這個總是會拋出一個 InvalidOperationException異常的應用爲例。我們按照如下的形式調用ApplicationBuilder的擴展方法UseExceptionHandler註冊了上述的這個ExceptionHandlerMiddleware中間件。這個擴展方法具有一個ExceptionHandlerOptions類型的參數,它的ExceptionHandler屬性返回的就是這個作爲異常處理器的RequestDelegate對象。

 public class Program
 {
 public static void Main()
 {
 RequestDelegate handler = async context => await context.Response.WriteAsync("Unhandled exception occurred!");
 
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app.UseExceptionHandler(new ExceptionHandlerOptions { ExceptionHandler = handler})
 .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
 .Build()
 .Run();
 }
 }

如上面的代碼片段所示,這個作爲異常處理器的RequestDelegate僅僅是將一個簡單的錯誤消息(“Unhandled exception occurred!”)作爲響應的內容。當我們利用瀏覽器訪問該應用的時候,這個定製的錯誤消息將會以如圖4所示的形式直接呈現在瀏覽器上。

最終作爲異常處理器的是一個類型爲RequestDelegate的委託對象,而ApplicationBuilder具有創建這個委託對象的能力。具體來說,我們可以根據異常處理的需要將相應的中間件註冊到某個ApplicationBuilder對象上,並最終利用這個ApplicationBuilder根據註冊的中間件創建出作爲異常處理器的RequestDelegate對象。 如果異常處理需要通過一個或者多箇中間件來完成,我們可以按照如下的形式調用另一個UseExceptionHandler方法重載。這個方法的參數類型爲Action<IApplicationBuilder>,我們調用它的Run方法註冊了一箇中間件來響應一個簡單的錯誤消息。

 public class Program
 {
 public static void Main()
 { 
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app.UseExceptionHandler(builder=>builder.Run(async context => await context.Response.WriteAsync("Unhandled exception occurred!")))
 .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
 .Build()
 .Run();
 }
 }

上面這兩種異常處理的形式都體現在提供一個RequestDelegate的委託對象來處理拋出的異常並完成最終的響應。如果應用已經設置了一個錯誤頁面,並且這個錯誤頁面具有一個固定的路徑,那麼我們在進行異常處理的時候就沒有必要提供這個RequestDelegate對象,而只需要重定向到錯誤頁面指向的路徑即可。這種採用服務端重定向的異常處理方式可以採用如下的形式調用另一個UseExceptionHandler方法重載來完成,這個方法的參數表示的就是重定向的目標路徑(“/error”),我們針對這個路徑註冊了一個路由來響應定製的錯誤消息。

 public class Program
 {
 public static void Main()
 {
 new WebHostBuilder()
 .UseKestrel()
 .ConfigureServices(svcs=>svcs.AddRouting())
 .Configure(app => app
  .UseExceptionHandler("/error")
  .UseRouter(builder=>builder.MapRoute("error", async context => await context.Response.WriteAsync("Unhandled exception occurred!")))
  .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
 .Build()
 .Run();
 }
 }

三、針對響應狀態碼定製錯誤頁面

由於Web應用採用HTTP通信協議,所以我們應該儘可能低迎合HTTP標準並將定義在協議規範中的語義應用到應用中。對於異常或者錯誤的語義表達在HTTP協議層面主要體現在響應報文的狀態碼上,具體來說HTTP通信的錯誤大體分爲如下兩種類型:

  • 客戶端錯誤:表示因客戶端提供不正確的請求信息而導致服務器不能正常處理請求,響應狀態碼範圍在400~499之間。
  • 服務端錯誤:表示服務器在處理請求過程中因自身的問題而發生錯誤,響應狀態碼在500~509之間。

正是因爲響應狀態碼是對錯誤或者異常語義最重要的表達,所以在很多情況下我們需要針對不同的響應狀態碼來定製顯示的錯誤信息。針對響應狀態碼對錯誤頁面的定製可以藉助一個類型爲StatusCodePagesMiddleware的中間件來實現,我們可以調用ApplicationBuilder相應的擴展方法來註冊這個中間件。

DeveloperExceptionPageMiddleware和ExceptionHandlerMiddleware中間件都是在後續請求處理過程中拋出異常的情況下才會被調用,而StatusCodePagesMiddleware被調用的前提是後續請求助理過程中產生一個錯誤響應狀態碼(範圍在400~599之間)。如果僅僅希望顯示一個統一的錯誤頁面,我們可以按照如下的形式調用擴展方法UseStatusCodePages註冊這個中間件,傳入該方法的兩個參數分別表示響應採用的媒體類型和主體內容。

 public class Program
 {
 public static void Main()
 { 
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app=>app
  .UseStatusCodePages("text/plain", "Error occurred ({0})")
  .Run(context=> Task.Run(()=>context.Response.StatusCode = 500)))
 .Build()
 .Run();
 }
 }

如上面的代碼片段所示,應用在處理請求的時候總是會將響應狀態碼設置爲500,所以最終的響應內容將由註冊的StatusCodePagesMiddleware中間件來提供。我們調用UseStatusCodePages方法的時候將響應的媒體類型設置爲“text/plain”,並將一段簡單的錯誤消息作爲了響應的主體內容。值得一提的時候,作爲響應內容的字符串可以包含一個佔位符({0}),StatusCodePagesMiddleware中間件最終會採用當前響應狀態碼來替換它。如果我們利用瀏覽器來訪問這個應用,將會得到如下圖所示的錯誤頁面。

如果我們希望針對不同的錯誤狀態碼顯示不同的錯誤頁面,那麼我們就需要將具體的請求處理邏輯實現在一個的狀態碼錯誤處理器中,並最終提供給StatusCodePagesMiddleware中間件。這個所謂的狀態碼錯誤處理器體現爲一個類型爲Func<StatusCodeContext, Task>的委託對象,作爲輸入的StatusCodeContext對象是對當前HttpContext的封裝,同時承載着其他一些與錯誤處理相關的選項設置,我們將在本系列後續部分對這個類型進行詳細介紹。

對於如下這個應用來說,它在處理任意一個請求是總是會隨機地選擇一個400~599之間的整數作爲響應的狀態碼,所以客戶端返回的響應內容總是通過註冊的StatusCodePagesMiddleware中間件來提供。我們在調用另一個UseStatusCodePages方法重載的時候,爲註冊的中間件指定了一個Func<StatusCodeContext, Task>對象作爲狀態碼錯誤處理器。

 public class Program
 {
 private static Random _random = new Random();
 
 public static void Main()
 {
 Func<StatusCodeContext, Task> handler = async context => {
 var response = context.HttpContext.Response;
 if (response.StatusCode < 500)
 {
  await response.WriteAsync($"Client error ({response.StatusCode})");
 }
 else
 {
  await response.WriteAsync($"Server error ({response.StatusCode})");
 }
 };
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app
  .UseStatusCodePages(handler)
  .Run(context => Task.Run(() => context.Response.StatusCode = _random.Next(400,599))))
 .Build()
 .Run();
 }
 }

我們指定的狀態碼錯誤處理器在處理請求的時候,根據響應狀態碼將錯誤分成客戶端錯誤和服務端錯誤兩種類型,並選擇針對性的錯誤消息作爲響應內容。當我們利用瀏覽器訪問這個應用的時候,顯示的錯誤消息將由響應狀態碼來決定。

在ASP.NET Core的世界裏,針對請求的處理總是體現爲一個RequestDelegate對象。如果請求的處理需要藉助一個或者多箇中間件來完成,我們可以將它們註冊到ApplicationBuilder對象上並利用它將中間件管道轉換成一個RequestDelegate對象。用於註冊StatusCodePagesMiddleware中間件的UseStatusCodePage方法還具有另一個重載,它允許我們採用這種方式來創建一個RequestDelegate對象來完成最終的請求處理工作,所以上面演示的這個應用完全可以改寫成如下的形式。

 public class Program
 {
 private static Random _random = new Random();
 public static void Main()
 {
 RequestDelegate handler = async context =>
 {
 var response = context.Response;
 if (response.StatusCode < 500)
 {
  await response.WriteAsync($"Client error ({response.StatusCode})");
 }
 else
 {
  await response.WriteAsync($"Server error ({response.StatusCode})");
 }
 };
 new WebHostBuilder()
 .UseKestrel()
 .Configure(app => app
  .UseStatusCodePages(builder=>builder.Run(handler))
  .Run(context => Task.Run(() => context.Response.StatusCode = _random.Next(400, 599))))
 .Build()
 .Run();
 }
 }

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對神馬文庫的支持。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章