由于ASP.NET Core应用是一个同时处理多个请求的服务器应用,所以在处理某个请求过程中抛出的异常并不会导致整个应用的终止。出于安全方面的考量,为了避免敏感信息的外泄,客户端在默认的情况下并不会得到详细的出错信息,这无疑会在开发环境下增加查错纠错的难度。对于生产环境来说,我们也希望最终用户能够根据具体的错误类型得到具有针对性并且友好的错误消息。ASP.NET Core提供了相应的中间件帮助我们将定制化的错误信息呈现出来,这些中间件都定义在“Microsoft.AspNetCore.Diagnostics”这个NuGet包中。在着重介绍这些中间件之前,我们照理演示几个简单的实例让读者朋友们对这些中间件的作用有一个大概的了解。[本文已经同步到《ASP.NET Core框架揭秘》之中]

目录
一、显示开发者异常页面
二、显示定制异常页面
三、针对响应状态码定制错误页面

一、显示开发者异常页面

一般情况下,如果ASP.NET Core在处理某个请求时出现异常,它一般会返回一个状态码为“500 Internal Server Error”的响应。为了避免一些敏感信息的外泄,详细的错误信息并不会随着响应发送给客户端,所以客户端只会得到一个很一般化的错误消息。以如下这个程序为例,服务端在处理每个请求时都会抛出一个类型为InvalidOperationException的异常。

   1: public class Program   2: {   3:     public static void Main()   4:     {   5:         new WebHostBuilder()   6:             .UseKestrel()   7:             .Configure(app => app.Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))   8:             .Build()   9:             .Run();  10:     }  11: }

当我们利用浏览器访问这个应用的时候,总是会得到如下图所示的这个错误页面。可以看出这个页面仅仅告诉我们目标应用当前无法正常处理本次请求,除了提供的响应状态码(“HTTP ERROR 500”)之外,它并没有提供任何有益于差错纠错的错误信息。

那么有人可能会觉得虽然浏览器上没有显示出任何详细的错误信息,也许它会隐藏在接收到的HTTP响应报文中。针对通过浏览器放出的这个请求,得到的响应内容如下所示,我们会发现响应报文根本没有主体部分,有限的几个报头也并没有承载任何与错误有关的信息。

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

由于应用并没有中断,浏览器上也并没有显示任何具有针对性的错误信息,开发人员在进行查错纠错的时候如何准确定位到作为错误根源的那一行代码呢?具体来说,我们又两种解决方案,一种就是利用日志,因为ASP.NET Core在进行请求处理时出现的任何错误都会被写入日志,所以我们可以通过注册相应的LoggerProvider(比如注册一个ConsoleLoggerProvider将日志直接写入宿主应用的控制台)到来获取写入的错误日志。

至于另一种解决方案,就是直接显示一个包含错误相应信息的错误页面,由于这个页面是在开发环境给开发者看的,所以我们将这个页面称为“开发者异常页面(Developer Exception Page)”。针对页面的自动呈现是利用一个名为DeveloperExceptionPageMiddleware的中间件来完成的,我们可以调用ApplicationBuilder的扩展方法UseDeveloperExceptionPage来注册这个中间件。

   1: public class Program   2: {   3:     public static void Main()   4:     {   5:         new WebHostBuilder()   6:             .UseKestrel()   7:             .Configure(app => app   8:                 .UseDeveloperExceptionPage()   9:                 .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))  10:             .Build()  11:             .Run();  12: }  13: }

一旦注册了这个DeveloperExceptionPageMiddleware中间件,ASP.NET Core应用在处理请求出现的异常信息就会以下图的形式直接出现在浏览器上,我们可以在这个页面中看到几乎所有的错误信息,包括异常的类型、消息和堆栈信息等。

开发者异常页面除了显示与抛出的异常相关的信息之外,还会以如下图所示的形式显示与当前请求上下文相关的信息,其中包括当前请求URL携带的所有查询字符串、所有请求报头以及Cookie的内容。如此详尽的信息无疑会极大地帮助开发人员尽快地找出错误的根源。

通过DeveloperExceptionPageMiddleware中间件呈现的错误页面仅仅是供开发人员使用的,详细的错误信息往往会携带一些敏感的信息,所以务必记住只有在开发环境才能注册这个中间件,如下所示的代码片段体现了针对DeveloperExceptionPageMiddleware中间件正确的注册方式。

   1: new WebHostBuilder()   2:     .UseStartup()   3:     …   4:     5: public class Startup   6: {   7:     public void Configure(IApplicationBuilder app, IHostingEnvironment env)   8:     {   9:         if (env.IsDevelopment())  10:         {  11:             app.UseDeveloperExceptionPage();  12:         }  13:     }  14: }

二、显示定制异常页面

DeveloperExceptionPageMiddleware中间件通过将异常详细信息和基于当前请求的内容直接呈现在错误页面中,这为开发人员的纠错诊断提供了极大的便利。但是在生产环境下,我们倾向于为最终的用户呈现一个定制的错误页面,而这可以通过注册另一个名为ExceptionHandlerMiddleware的中间件来实现。顾名思义,这个中间件旨在提供一个异常处理器(Exception Handler)来处理抛出的异常。实际上这个所谓的异常处理器就是一个类型为RequestDelegate的委托对象,ExceptionHandlerMiddleware中间件捕捉到抛出的异常后利用它来响应当前的请求。

还是以上面创建的这个总是会抛出一个 InvalidOperationException异常的应用为例。我们按照如下的形式调用ApplicationBuilder的扩展方法UseExceptionHandler注册了上述的这个ExceptionHandlerMiddleware中间件。这个扩展方法具有一个ExceptionHandlerOptions类型的参数,它的ExceptionHandler属性返回的就是这个作为异常处理器的RequestDelegate对象。

   1: public class Program
   2: {
   3:     public static void Main()
   4:     {
   5:         RequestDelegate handler = async context => await context.Response.WriteAsync("Unhandled exception occurred!");
   6: 
   7:         new WebHostBuilder()
   8:             .UseKestrel()
   9:             .Configure(app => app.UseExceptionHandler(new ExceptionHandlerOptions { ExceptionHandler = handler})
  10:             .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))
  11:             .Build()
  12:             .Run();
  13:     }
  14: }

如上面的代码片段所示,这个作为异常处理器的RequestDelegate仅仅是将一个简单的错误消息(“Unhandled exception occurred!”)作为响应的内容。当我们利用浏览器访问该应用的时候,这个定制的错误消息将会以如图4所示的形式直接呈现在浏览器上。

最终作为异常处理器的是一个类型为RequestDelegate的委托对象,而ApplicationBuilder具有创建这个委托对象的能力。具体来说,我们可以根据异常处理的需要将相应的中间件注册到某个ApplicationBuilder对象上,并最终利用这个ApplicationBuilder根据注册的中间件创建出作为异常处理器的RequestDelegate对象。如果异常处理需要通过一个或者多个中间件来完成,我们可以按照如下的形式调用另一个UseExceptionHandler方法重载。这个方法的参数类型为Action,我们调用它的Run方法注册了一个中间件来响应一个简单的错误消息。

   1: public class Program   2: {   3:     public static void Main()   4:     {           5:         new WebHostBuilder()   6:             .UseKestrel()   7:             .Configure(app => app.UseExceptionHandler(builder=>builder.Run(async context => await context.Response.WriteAsync("Unhandled exception occurred!")))   8:             .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))   9:         .Build()  10:         .Run();  11:     }  12: }

上面这两种异常处理的形式都体现在提供一个RequestDelegate的委托对象来处理抛出的异常并完成最终的响应。如果应用已经设置了一个错误页面,并且这个错误页面具有一个固定的路径,那么我们在进行异常处理的时候就没有必要提供这个RequestDelegate对象,而只需要重定向到错误页面指向的路径即可。这种采用服务端重定向的异常处理方式可以采用如下的形式调用另一个UseExceptionHandler方法重载来完成,这个方法的参数表示的就是重定向的目标路径(“/error”),我们针对这个路径注册了一个路由来响应定制的错误消息。

   1: public class Program   2: {   3:     public static void Main()   4:     {   5:         new WebHostBuilder()   6:             .UseKestrel()   7:             .ConfigureServices(svcs=>svcs.AddRouting())   8:             .Configure(app => app   9:                 .UseExceptionHandler("/error")  10:                 .UseRouter(builder=>builder.MapRoute("error", async context => await context.Response.WriteAsync("Unhandled exception occurred!")))  11:                 .Run(context => Task.FromException(new InvalidOperationException("Manually thrown exception..."))))  12:         .Build()  13:         .Run();  14:     }  15: }

三、针对响应状态码定制错误页面

由于Web应用采用HTTP通信协议,所以我们应该尽可能低迎合HTTP标准并将定义在协议规范中的语义应用到应用中。对于异常或者错误的语义表达在HTTP协议层面主要体现在响应报文的状态码上,具体来说HTTP通信的错误大体分为如下两种类型:

  • 客户端错误:表示因客户端提供不正确的请求信息而导致服务器不能正常处理请求,响应状态码范围在400~499之间。

  • 服务端错误:表示服务器在处理请求过程中因自身的问题而发生错误,响应状态码在500~509之间。

正是因为响应状态码是对错误或者异常语义最重要的表达,所以在很多情况下我们需要针对不同的响应状态码来定制显示的错误信息。针对响应状态码对错误页面的定制可以借助一个类型为StatusCodePagesMiddleware的中间件来实现,我们可以调用ApplicationBuilder相应的扩展方法来注册这个中间件。

DeveloperExceptionPageMiddleware和ExceptionHandlerMiddleware中间件都是在后续请求处理过程中抛出异常的情况下才会被调用,而StatusCodePagesMiddleware被调用的前提是后续请求助理过程中产生一个错误响应状态码(范围在400~599之间)。如果仅仅希望显示一个统一的错误页面,我们可以按照如下的形式调用扩展方法UseStatusCodePages注册这个中间件,传入该方法的两个参数分别表示响应采用的媒体类型和主体内容。

   1: public class Program   2: {   3:     public static void Main()   4:     {           5:         new WebHostBuilder()   6:             .UseKestrel()   7:             .Configure(app=>app   8:                 .UseStatusCodePages("text/plain", "Error occurred ({0})")   9:                 .Run(context=> Task.Run(()=>context.Response.StatusCode = 500)))  10:         .Build()  11:         .Run();  12:     }  13: }

如上面的代码片段所示,应用在处理请求的时候总是会将响应状态码设置为500,所以最终的响应内容将由注册的StatusCodePagesMiddleware中间件来提供。我们调用UseStatusCodePages方法的时候将响应的媒体类型设置为“text/plain”,并将一段简单的错误消息作为了响应的主体内容。值得一提的时候,作为响应内容的字符串可以包含一个占位符({0}),StatusCodePagesMiddleware中间件最终会采用当前响应状态码来替换它。如果我们利用浏览器来访问这个应用,将会得到如下图所示的错误页面。

如果我们希望针对不同的错误状态码显示不同的错误页面,那么我们就需要将具体的请求处理逻辑实现在一个的状态码错误处理器中,并最终提供给StatusCodePagesMiddleware中间件。这个所谓的状态码错误处理器体现为一个类型为Func的委托对象,作为输入的StatusCodeContext对象是对当前HttpContext的封装,同时承载着其他一些与错误处理相关的选项设置,我们将在本系列后续部分对这个类型进行详细介绍。

对于如下这个应用来说,它在处理任意一个请求是总是会随机地选择一个400~599之间的整数作为响应的状态码,所以客户端返回的响应内容总是通过注册的StatusCodePagesMiddleware中间件来提供。我们在调用另一个UseStatusCodePages方法重载的时候,为注册的中间件指定了一个Func对象作为状态码错误处理器。

   1: public class Program   2: {   3: private static Random _random = new Random();   4:     5:     public static void Main()   6:     {   7:         Func handler = async context => {   8:             var response = context.HttpContext.Response;   9:             if (response.StatusCode < 500)  10:             {  11:                 await response.WriteAsync($"Client error ({response.StatusCode})");  12:             }  13:             else  14:             {  15:                 await response.WriteAsync($"Server error ({response.StatusCode})");  16:             }  17:         };  18:         new WebHostBuilder()  19:             .UseKestrel()  20:             .Configure(app => app  21:                 .UseStatusCodePages(handler)  22:                 .Run(context => Task.Run(() => context.Response.StatusCode = _random.Next(400,599))))  23:             .Build()  24:             .Run();  25:     }  26: }

我们指定的状态码错误处理器在处理请求的时候,根据响应状态码将错误分成客户端错误和服务端错误两种类型,并选择针对性的错误消息作为响应内容。当我们利用浏览器访问这个应用的时候,显示的错误消息将由响应状态码来决定。

在ASP.NET Core的世界里,针对请求的处理总是体现为一个RequestDelegate对象。如果请求的处理需要借助一个或者多个中间件来完成,我们可以将它们注册到ApplicationBuilder对象上并利用它将中间件管道转换成一个RequestDelegate对象。用于注册StatusCodePagesMiddleware中间件的UseStatusCodePage方法还具有另一个重载,它允许我们采用这种方式来创建一个RequestDelegate对象来完成最终的请求处理工作,所以上面演示的这个应用完全可以改写成如下的形式。

   1: public class Program   2: {   3:     private static Random _random = new Random();   4:     public static void Main()   5:     {   6:         RequestDelegate handler = async context =>   7:         {   8:             var response = context.Response;   9:             if (response.StatusCode < 500)  10:             {  11:                 await response.WriteAsync($"Client error ({response.StatusCode})");  12:             }  13:             else  14:             {  15:                 await response.WriteAsync($"Server error ({response.StatusCode})");  16:             }  17:         };  18:         new WebHostBuilder()  19:             .UseKestrel()  20:             .Configure(app => app  21:                 .UseStatusCodePages(builder=>builder.Run(handler))  22:                 .Run(context => Task.Run(() => context.Response.StatusCode = _random.Next(400, 599))))  23:             .Build()  24:             .Run();  25:     }  26: }

asp 开发app_ASP.NET Core应用的错误处理[1]:三种呈现错误页面的方式相关推荐

  1. asp.net错误处理的三种方法

    asp.net错误处理的三种方法 以前有过一篇博文,主要是出错时调至错误页面,然后将当前出错的页面存入Session,然后执行刷新等操作,该机制主要用于 程序员自己控制错误,但是有些错误和异常是程序员 ...

  2. c语言错误1004,excel宏运行时提示错误1004的三种解决方法

    在使用excel宏功能的时候,一些网友会遇到excel宏运行不了,软件提示:"运行时错误1004,应用程序定义或对象定义错误"的问题,那么,excel宏运行时错误1004怎么办?幸 ...

  3. “fatal error C1010”错误解决的三种方法

    "fatal error C1010"错误解决的三种方法 参考文章: (1)"fatal error C1010"错误解决的三种方法 (2)https://ww ...

  4. ETSI TR101 290监测的三种级别错误接收端现象

    对于码流分析仪所提供ETSI TR101 290监测的三种级别错误,接收端将会出现如下现象. 级别 错误类型 接收端现象 一 级 错 误 同步丢失错 黑屏.静帧和马赛克.画面不流畅现象 同步字节错 黑 ...

  5. 出现No persister for错误后有如下三种解决办法

    出现No persister for错误后有如下三种解决办法: 1.检查hbm.xml文件,配置文件后缀名写错 mapping file 必须是.hbm.xml结尾. 2.hibernate 的map ...

  6. 《ASP.NET Core 6框架揭秘》实例演示[32]:错误页面的N种呈现方式

    由于ASP.NET是一个同时处理多个请求的Web应用框架,所以在处理某个请求过程中出现异常并不会导致整个应用的中止.出于安全方面的考量,为了避免敏感信息外泄,客户端在默认情况下并不会得到详细的出错信息 ...

  7. .Net Core 实现WebSocket Server 的另外三种方式

    回顾 之前已经写过关于<WebSocket 原生socket实现>和<.Net Core WebSocket 服务端与客户端完整示例>以及<基于.Net TcpListe ...

  8. EF Core中避免贫血模型的三种行之有效的方法(翻译)

    [Paul Hiles: 3 ways to avoid an anemic domain model in EF Core :https://www.devtrends.co.uk/blog/3-w ...

  9. 【Code】代码答案错误怎么办?三种方法教你如何查错

    查错的方法 1.目测法 1 首先你要找到一组使你程序错误的数据. 2 然后把你的大脑想象成电脑. 3一步一步执行程序中的语句. 4 比较你的输出结果和标准输出结果是否相同. 2.调试法 其实很简单,就 ...

最新文章

  1. R语言使用ggpubr包可视化分组条形图(grouped bar plot)、并添加误差条(error bar、误差条显示平均值+/-sd)、添加分组填充色、添加jitter、自定义分组抖动数据点形状
  2. 深度学习(五)caffe环境搭建
  3. STM32之独立版USB(Host)驱动+MSC+Fatfs移植
  4. 运营商主动关闭WLAN热点背后:关键是找准产业位置
  5. win7屏保时间设置_论如何优雅的设置手机和电脑时间屏保!
  6. linux服务器安装caffe(无root权限)
  7. typedef 浅析
  8. 真空压力变送器怎么样零点标定_对于变送器你知道多少?
  9. docker java 最小镜像_Docker多步构建更小的Java镜像
  10. 数据库变为可疑_数据库显示可疑的修复方法
  11. 网件使用计算机mac地址吗,网件路由器怎么设置mac地址过滤
  12. 数据结构与算法(回文数与括号匹配问题)
  13. 根据IP地址查询所在地
  14. 华为云服务器配置教程
  15. python之html网页转PDF
  16. 使用nw.js将网址打包生成exe可安装程序支持xp系统
  17. 皮革行业经销商在线系统:简化采购审批流程,轻松管控代理商
  18. 怎么安装LINUX系统
  19. 计算机错误651是什么故障,错误651是什么意思
  20. 一张图揭秘在阿里、腾讯、美团工作的区别

热门文章

  1. java webservice 服务器_java如何写webservice服务端
  2. 端口可以随便设置吗_驱动可以随便更新吗?
  3. centos 7 ssh 安装mysql,Linux服务器远程ssh为centos7安装MySQL
  4. 安利几个优质NLP开源项目!搜索、问答、情感分析…
  5. 从前,有只小仙女叫...
  6. 深入理解 Objective-C:方法缓存
  7. Java动态追踪技术探究 1
  8. 论文浅尝 | Data Intelligence第4期正式上线啦
  9. 会议交流 | 智能风控技术峰会(请关注图分析相关论坛)
  10. python 异常处理相关说明