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

由于ASP.NET是一个同时处理多个请求的Web应用框架,所以在处理某个请求过程中出现异常并不会导致整个应用的中止。出于安全方面的考量,为了避免敏感信息外泄,客户端在默认情况下并不会得到详细的出错信息,这无疑会在开发过程中增加查错和纠错的难度。对于生产环境来说,我们也希望最终用户能够根据具体的错误类型得到具有针对性并且友好的错误消息。ASP.NET提供的相应的中间件可以帮助我们将定制化的错误信息呈现出来。[本文节选《ASP.NET Core 6框架揭秘》第21章]

目录
[2101]开发者异常页面的呈现(源代码)
[2102]定制异常页面的呈现(源代码)
[2103]利用注册的中间件处理异常(源代码)
[2104]针对异常页面的重定向(源代码)
[2105]基于响应状态码错误页面的呈现(设置响应内容模板)(源代码)
[2106]基于响应状态码错误页面的呈现(提供异常处理器)(源代码)
[2107]基于响应状态码错误页面的呈现(利用中间件创建异常处理器)(源代码)

[2101]开发者异常页面的呈现

如果ASP.NET应用在处理某个请求时出现异常,它一般会返回一个状态码为“500 Internal Server Error”的响应。为了避免一些敏感信息的外泄,客户端只会得到一个很泛化的错误消息。以如下所示的程序为例,处理根路径的请求时都会抛出一个InvalidOperationException类型的异常。

var app = WebApplication.Create();
app.MapGet("/",
void () => throw new InvalidOperationException("Manually thrown exception"));
app.Run();

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

6cddf0faa8a6ba85caa4f9d8b4bff1c3.png

图1 默认的错误页面

有人认为浏览器上虽然没有显示任何详细的错误信息,但这并不意味着HTTP响应报文中也没有携带任何详细的出错信息。如下所示的服务端会返回的HTTP响应报文,该响应没有主体内容,有限的几个报头也并没有承载任何与错误有关的信息。

HTTP/1.1 500 Internal Server Error
Content-Length: 0
Date: Sun, 07 Nov 2021 08:34:18 GMT
Server: Kestrel

由于应用并没有中断,浏览器上也并没有显示任何具有针对性的错误信息,我们无法知道背后究竟出现了什么错误。这个问题有两种解决方案:一种是利用日志,ASP.NET在处理请求过程中出现异常时,会发出相应的日志事件,我们可以注册相应的ILoggerProvider对象将日志输出到指定的渠道。另一种解决方案就是利用注册的DeveloperExceptionPageMiddleware中间件显示一个“开发者异常页面(Developer Exception Page)”。

如下的演示程序调用IApplicationBuilder接口的UseDeveloperExceptionPage扩展方法来注册了这个中间件。该程序注册了一个路由模板为“{foo}/{bar}”的终结点,后者在处理请求时直接抛出异常。

var app = WebApplication.Create();
app.UseDeveloperExceptionPage();
app.MapGet("{foo}/{bar}",
void () => throw new InvalidOperationException("Manually thrown exception"));
app.Run();

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

657d834620c5c1a0e3d2ea21c856b116.png

图2 开发者异常页面(基本信息)

开发者异常页面除了显示与抛出的异常相关的信息,还会以图3所示的形式显示与当前请求上下文相关的信息,包括当前请求URL携带的所有查询字符串、所有请求报头、Cookie的内容和路由信息(终结点和路由参数)。如此详尽的信息无疑会极大地帮助开发人员尽快找出错误的根源。由于此页面上往往会携带一些敏感的信息,所以只有在开发环境才能注册这个中间件。实际上Minimal API在开发环境会默认注册这个中间件。

341817f0802a7e2a097816697bb1a735.png

图3 开发者异常页面(详细信息)

[2102]定制异常页面的呈现

由于ExceptionHandlerMiddleware中间件直接利用提供的RequestDelegate委托来处理出现异常的请求,我们可以利用它呈现一个定制化的错误页面。如下的演示程序通过调用IApplicationBuilder接口的UseExceptionHandler扩展方法注册了这个中间件,提供的的ExceptionHandlerOptions配置选项指定了一个指向HandleErrorAsync方法的RequestDelegate委托作为异常处理器。

var options = new ExceptionHandlerOptions { ExceptionHandler = HandleErrorAsync };
var app = WebApplication.Create();
app.UseExceptionHandler(options);
app.MapGet("/",
void () 
=> throw new InvalidOperationException("Manually thrown exception"));
app.Run();static Task HandleErrorAsync(HttpContext context) 
=> context.Response.WriteAsync("Unhandled exception occurred!");

如上面的代码片段所示,HandleErrorAsync方法仅仅是将一个简单的错误消息(Unhandled exception occurred!)作为响应的内容。演示程序注册了一个针对根路径(“/”)的并且直接抛出异常的终结点,当我们利用浏览器访问该终结点时,这个定制的错误消息会以图4所示的形式直接呈现在浏览器上。

ece2abfd45ef6b316553531fc11c8a56.png

图4 定制的错误页面

[2103]利用注册的中间件处理异常

由于ExceptionHandlerMiddleware中间件的异常处理器的是一个RequestDelegate委托,而IApplicationBuilder对象具有利用注册的中间件来创建这个委托对象的能力,所以用于注册该中间件的UseExceptionHandler扩展方法提供了一个参数类型为Action<IApplicationBuilder>重载。如下的演示程序调用了这个方法,在提供的作为参数的Action<IApplicationBuilder>委托中,我们调用了IApplicationBuilder接口的Run方法注册了一个中间件来处理异常,访问启动后的程序同样会得到如图21-4的错误信息(S2103)。

var app = WebApplication.Create();
app.UseExceptionHandler(app2 => app2.Run(HandleErrorAsync))
app.MapGet("/",
void () 
=> throw new InvalidOperationException("Manually thrown exception"));
app.Run();static Task HandleErrorAsync(HttpContext context)  
=> context.Response.WriteAsync("Unhandled exception occurred!");

[2104]针对异常页面的重定向

如果应用已经提供了一个错误页面,ExceptionHandlerMiddleware中间件在进行异常处理时可以直接重定向到该页面就可以了。如下的演示程序采用这种方式调用了另一个UseExceptionHandler扩展方法重载,作为参数的字符串(“/error”)指定的就是错误页面的路径,访问启动后的程序同样会得到如图4的错误信息。

var app = WebApplication.Create();
app.UseExceptionHandler("/error");
app.MapGet("/",
void () 
=> throw new InvalidOperationException("Manually thrown exception"));
app.MapGet("/error", HandleErrorAsync);
app.Run();static Task HandleErrorAsync(HttpContext context)  
=> context.Response.WriteAsync("Unhandled exception occurred!");

[2105]基于响应状态码错误页面的呈现(设置响应内容模板)

我们知道HTTP语义中的错误是由响应的状态码来表达的,涉及的错误大体划分为如下两种类型:

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

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

StatusCodePagesMiddleware中间件帮助我们针对响应状态码对错误页面进行定制。该中间件只有在后续管道产生一个错误响应状态码(范围为400~599)才会将错误页面呈现出来。如下的演示程序通过调用UseStatusCodePages扩展方法注册了这个中间件,作为参数的两个字符串分别是响应的媒体类型和作为主体内容的模板,占位符“{0}”将被状态码进行填充。

var app = WebApplication.Create();
app.UseStatusCodePages("text/plain", "Error occurred ({0})");
app.MapGet("/", void (HttpResponse response) => response.StatusCode = 500);
app.Run();

我们针对根路径(“/”)注册了一个终结点,后者在处理请求时直接返回状态码为500的响应。应用启动后,针对该路径请求将会得到如图5所示的错误页面。

2306d634e74177f4dfe4f60aeb896efe.png

图5 针对错误响应状态码定制的错误页面

[2106]基于响应状态码错误页面的呈现(提供异常处理器)

StatusCodePagesMiddleware中间件的错误处理器体现为一个Func<StatusCodeContext, Task>委托,作为输入的StatusCodeContext是对当前HttpContext上下文的封装。如下的演示程序定义了一个与此委托具有一致声明的HandleErrorAsync来呈现错误页面,UseStatusCodePages扩展方法指定的Func<StatusCodeContext, Task>委托指向这个方法。

using Microsoft.AspNetCore.Diagnostics;
var random = new Random();
var app = WebApplication.Create();
app.UseStatusCodePages(HandleErrorAsync);
app.MapGet("/", void (HttpResponse response) 
=> response.StatusCode = random.Next(400,599));
app.Run();static  Task HandleErrorAsync(StatusCodeContext context)
{var response = context.HttpContext.Response;return response.StatusCode < 500? response.WriteAsync($"Client error ({response.StatusCode})"): response.WriteAsync($"Server error ({response.StatusCode})");
}

我们针对根路径(“/”)注册的终结点会随机返回一个状态码在(400,599)区间内的响应。用来处理错误的HandleErrorAsync方法会根据状态码所在的区间(400~499, 500~599)分别显式“Client error”和“Server error”。应用启动后,针对根路径的请求会得到如图6所示错误页面。

53fdbf630d36778c5055733299e78ec6.png

图6 针对错误响应状态码定制的错误页面

[2107]基于响应状态码错误页面的呈现(利用中间件创建异常处理器)

在ASP.NET的世界里,针对请求的处理总是体现为一个RequestDelegate委托,而IApplicationBuilder对象具有根据注册的中间件构建这个委托的能力,所以 UseStatusCodePages方法还具有另一个将Action<IApplicationBuilder>委托作为参数的重载。如下的演示程序调用了这个重载,我们利用提供的委托调用了IApplicationBuilder对象的Run扩展方法注册了一个中间件来处理异常(S2107)。

var random = new Random();
var app = WebApplication.Create();
app.UseStatusCodePages(app2 => app2.Run(HandleErrorAsync));
app.MapGet("/", void (HttpResponse response) => response.StatusCode = random.Next(400,599));
app.Run();static  Task HandleErrorAsync(HttpContext context)
{var response = context.Response;return response.StatusCode < 500? response.WriteAsync($"Client error ({response.StatusCode})"): response.WriteAsync($"Server error ({response.StatusCode})");
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/281403.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringMVC接受JSON参数详解及常见错误总结我改

SpringMVC接受JSON参数详解及常见错误总结 最近一段时间不想使用Session了&#xff0c;想感受一下Token这样比较安全&#xff0c;稳健的方式&#xff0c;顺便写一个统一的接口给浏览器还有APP。所以把一个练手项目的前台全部改成Ajax了&#xff0c;跳转再使用SpringMVC控制转发…

软件定义存储的定制化怎么走?

引言 当前&#xff0c;软件定义存储成为业内超高速增长的典型。有研究人员称&#xff0c;从2014年到2019年&#xff0c;软件定义存储市场将从14亿美元增长到62亿美元以上&#xff0c;年复合增长率将达35%。软件定义存储所带来的优势显而易见&#xff0c;但是对于企业来说&#…

Golang并发模型:合理退出并发协程

goroutine作为Golang并发的核心&#xff0c;我们不仅要关注它们的创建和管理&#xff0c;当然还要关注如何合理的退出这些协程&#xff0c;不&#xff08;合理&#xff09;退出不然可能会造成阻塞、panic、程序行为异常、数据结果不正确等问题。这篇文章介绍&#xff0c;如何合…

剑网服务器维护,12月31日服务器例行维护公告

隐元秘鉴新增以下江湖行里使用趣味道具的成就&#xff1a;压酒唤客尝&#xff1a;使用压酒三十次欲解红烛意&#xff1a;使用烛影三十次闲情吹笛子&#xff1a;使用吹断三十次引弦中落雀&#xff1a;使用千鸟三十次黄云动风色&#xff1a;使用风色三十次卷抒平生意&#xff1a;…

一款 Windows 软件快捷助手

WPF 开发的 Windows 软件快捷助手Windows 软件快捷助手作者&#xff1a;WPFDevelopersOrg - 驚鏵原文链接&#xff1a;https://github.com/WPFDevelopersOrg/SoftwareHelper框架使用.NET40&#xff1b;Visual Studio 2019;项目使用 MIT 开源许可协议&#xff1b;项目使用 MVV…

关于8位AD_DA转换芯片的采样率问题

关于使用Keil计算程序执行时间 打开Keil程序&#xff0c;进入“启动/停止调试”界面。在需要暂停的地方设置断点&#xff08;在该句程序前双击&#xff09;。在程序上方有一行工具栏&#xff1a;此工具栏分别代表复位、运行、停止、步进、步越、步出、运行到光标处等。 点击运…

CYQ.Data 数据框架 V4.0 开源版本发布(源码提供下载,秋色园V2.5版本标配框架)

说明的说明&#xff1a; 博客园团队两次移此文出首页&#xff0c;说 这篇文章不属于知识分享型文章&#xff0c;并且有广告嫌疑。 本文的确属于分享型文章&#xff0c;而且分享的知识点比其它文章都多很多&#xff0c;看看网友回复“谢谢分享”就知道是分享型文章了。 所谓广告…

oracle 分组后取每组第一条数据

数据格式 分组取第一条的效果 sql SELECT * FROM (SELECT ROW_NUMBER() OVER(PARTITION BY x ORDER BY y DESC) rn, test1.* FROM test1) WHERE rn 1 ;

永大服务器进去显示字母,永大电梯服务器使用说明

永大电梯服务器使用说明2021-05-25一&#xff0e; 目的&#xff1a;用于工务交车前对MPU和XDR板进行调试。二&#xff0e; 对应作业&#xff1a;1-1对MPU电梯调试接线连接&#xff1a;1).对MPUGB2(A2)版电梯调试时(以及A2前版本)&#xff0c;接线情况如下&#xff1a;连接顺序连…

树莓派Zero 2 W(ubuntu-22.04)通过.NET6和libusb操作USB读写

有这个想法的初衷喜欢电子和DIY硬件的朋友对稚晖君应该都不陌生&#xff0c;他定期都会分享一些自己做的好玩的硬件&#xff0c;他之前做了一个ElectronBot桌面机器人我就很感兴趣&#xff0c;所以就自己也做了一个。起初我只是自己开发了一个叫电子脑壳的上位机软件&#xff0…

bzoj4589

fwt 原理并不知道 nim游戏石子异或和0后手赢 那么也就是求a[1]^a[2]^...^a[n]0的方案数 这个和bzoj3992一样可以dp dp[i][j]表示前i个数异或和为j的方案数 dp[0][0] 1 dp[i][j] dp[i - 1][k] * a[p] p ^ k j a[p] 0 / 1 表示有没有p这个数 这个东西也不能矩阵快速幂 但是我…

UnicodeDecodeError: 'ascii' codec can't decode byte 0xe5 in position 85

UnicodeDecodeError: ascii codec cant decode byte 0xe5 in position 85;import sys reload(sys) sys.setdefaultencoding(utf8)

JS设计模式五:职责链模式

职责链模式简述 职责连是由多个不同的对象组成的&#xff0c;有发送者跟接收者&#xff0c;分别负责信息的发送跟接收&#xff0c;其中&#xff0c;链中第一个对象是 职责连是由多个不同的对象组成的&#xff0c;发送者是发送请求的对象&#xff0c;接收者接收请求并且对其进行…

web框架之Django(一)

Python的WEB框架有Django、Tornado、Flask 等多种&#xff0c;Django相较与其他WEB框架其优势为&#xff1a;大而全&#xff0c;框架本身集成了ORM、模型绑定、模板引擎、缓存、Session等诸多功能。 基本配置 一、创建django程序 终端命令&#xff1a;django-admin startprojec…

写一个易于维护使用方便性能可靠的Hybrid框架(一)—— 思路构建

写一个易于维护使用方便性能可靠的Hybrid框架&#xff08;二&#xff09;—— 插件化 写一个易于维护使用方便性能可靠的Hybrid框架&#xff08;三&#xff09;—— 配置插件 前言 本来上一篇博文写完&#xff0c;我就告诉自己&#xff0c;这是最后一篇&#xff0c;之后不再总结…

程序员制作出价值5亿外卖神器却不能取消订单,你知道吗?

小编今日给大家带来RACDisopsable&#xff0c;大家可能有部分人对这个会感觉到很陌生&#xff0c;那么我就用一句话来表达就是他可以帮我们取消订阅。那么又会有人会对这个产生疑问了&#xff0c;我们什么时候需要用到这个取消订阅了打个实际的例子来说吧&#xff0c;今天我在饿…

Computer

链接&#xff1a;http://acm.hdu.edu.cn/showproblem.php?pid2196https://blog.csdn.net/shuangde800/article/details/9732825#include<iostream> #include<cstdio> #include<algorithm> #include<vector> #include<queue> #include<cmath&…

智慧“昆明”在路上 未来充满精彩

智慧城市是运用物联网、云计算、大数据、移动互联网、空间地理信息集成等新一代信息技术&#xff0c;促进城市规划、建设、管理和服务智慧化的新理念和新模式。近年来&#xff0c;昆明市全面加快智慧城市建设&#xff0c;力争通过三年的努力&#xff0c;打造区域信息辐射中心的…

《精读 Mastering ABP Framework》教程发布

精读《Mastering ABP Framework》学习总结&#xff0c;掌握软件开发最佳实践&#xff0c;构建可维护 .NET 解决方案。从 ABP Framework 框架中学习如何构建现代 WEB 应用程序。掌握 ABP Framework 框架ABP Framework 是一个完整的基础架构&#xff0c;遵循软件开发最佳实践&…

C# 委托知识总结

1.什么是委托&#xff0c;为什么要使用委托 我正在埋头苦写程序&#xff0c;突然想喝水&#xff0c;但是又不想自己去掉杯水而打断自己的思路&#xff0c;于是我就想让女朋友去给我倒水。她去给我倒水&#xff0c;首先我得让她知道我想让她干什么&#xff0c;通知她之后我可以继…