前言

          系统异常监控可以说是重中之重,系统不可能一直运行良好,开发和运维也不可能24小时盯着系统,系统抛异常后我们应当在第一时间收到异常信息。在Asp.net Core里我使用拦截器和中间件两种方式来监控异常。全局异常监控的数据最好还是写入数据库,方便查询。

配置NLog

QQ截图20191031103620.jpg

NLog配置文件

<?xml version="1.0" encoding="utf-8"?> <nlog xmlns="upload/201911011137598010.jpeg#align=left&display=inline&height=357&name=QQ%E6%88%AA%E5%9B%BE20191031101525.jpg&originHeight=357&originWidth=566&search=&size=13621&status=done&width=566" alt="QQ截图20191031101525.jpg" style="margin: 0px; padding: 0px; border: none; max-width: 900px; height: auto;" />
如果未使用全局异常捕获,则直接抛出如下异常
QQ截图20191031101516.jpg
         客户端抛出异常后,可查看磁盘写入日志,这里看到我关注的系统编号,主机ip,堆栈信息和异常描述信息。
QQ截图20191031160132.jpg

中间件

定义中间件,定义中间件时先导入日志命名空间Microsoft.Extensions.Logging。

public class GlobalExceptionMiddleware {     private readonly RequestDelegate next;     private ILogger<GlobalExceptionMiddleware> logger;     public GlobalExceptionMiddleware(RequestDelegate next, ILogger<GlobalExceptionMiddleware> logger)     {         this.next = next;         this.logger = logger;     }      public async Task Invoke(HttpContext context)     {         try         {             await next.Invoke(context);         }         catch (Exception ex)         {             await HandleExceptionAsync(context, ex);         }     }       private async Task HandleExceptionAsync(HttpContext context, Exception e)     {         if (e.GetType() == typeof(BusException))         {             //如果是自定义异常,则不做处理         }         else         {          }          //记日志          int sysId = 1;         string ip = context.Connection.RemoteIpAddress.ToString();         logger.LogError($"系统编号:{sysId},主机IP:{ip},堆栈信息:{e.StackTrace},异常描述:{e.Message}");         string result = System.Text.Json.JsonSerializer.Serialize(ResultBody.error(e.Message));         await context.Response.WriteAsync(result);     } }

在Startup.Configure方法里注册中间件。

public void Configure(IApplicationBuilder app, IWebHostEnvironment env,ILoggerFactory loggerFactory) {     if (env.IsDevelopment())     {         app.UseDeveloperExceptionPage();     }     else     {         app.UseExceptionHandler("/Home/Error");     }      //注册异常处理中间件     app.UseMiddleware<GlobalExceptionMiddleware>();      app.UseStaticFiles();      app.UseRouting();      app.UseAuthorization();      app.UseEndpoints(endpoints =>                      {                          endpoints.MapControllerRoute(                              name: "default",                              pattern: "{controller=Home}/{action=Index}/{id?}");                      }); }

中间件这里处理异常最后向客户端响应写入了一个字符串,这是个拦截器处理方式不同的地方。当然对客户端或者前端来说还是JSON对象更直观些。

参考链接

https://www.cnblogs.com/suizhikuo/p/8822352.html
https://www.cnblogs.com/viter/p/10013195.html
https://www.jianshu.com/p/cab597211136

博客地址: