前言

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

配置NLog

NLog配置文件

<?xml version="1.0" encoding="utf-8"?>
<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"autoReload="true"internalLogLevel="info"internalLogFile="d:\temp\internal-nlog.txt"><!-- the targets to write to --><targets><!-- write logs to file  --><target xsi:type="File" name="allfile" fileName="d:\temp\nlog-all-${shortdate}.log"layout="${longdate}|${event-properties:item=EventId.Id}|${uppercase:${level}}|${logger}|${message} ${exception}" /><!-- another file log, only own logs. Uses some ASP.NET core renderers --><target xsi:type="File" name="ownFile-web" fileName="d:\temp\nlog-own-${shortdate}.log"layout="${longdate}|${event-properties:item=EventId.Id}|${uppercase:${level}}|${logger}|${message} ${exception}|url: ${aspnet-request-url}|action: ${aspnet-mvc-action}" /><!-- write to the void aka just remove --><target xsi:type="Null" name="blackhole" /></targets><!-- rules to map from logger name to target --><rules><!--All logs, including from Microsoft--><logger name="*" minlevel="Trace" writeTo="allfile" /><!--Skip Microsoft logs and so log only own logs--><logger name="Microsoft.*" minlevel="Trace" writeTo="blackhole" final="true" /><logger name="*" minlevel="Trace" writeTo="ownFile-web" /></rules>
</nlog>

注入NLog

在Program.cs里注入NLog依赖,添加依赖前需要导入两个命名空间Microsoft.Extensions.Logging、 NLog.Web。

public class Program
{public static void Main(string[] args){CreateHostBuilder(args).Build().Run();}public static IHostBuilder CreateHostBuilder(string[] args) =>Host.CreateDefaultBuilder(args).ConfigureWebHostDefaults(webBuilder =>{webBuilder.UseStartup<Startup>();}).ConfigureLogging(logging=>{logging.ClearProviders();logging.SetMinimumLevel(Microsoft.Extensions.Logging.LogLevel.Trace);}).UseNLog();
}

拦截器

在Asp.Mvc里最常用的拦截器,在Asp.net Core里也是支持的。先定义拦截器,再注入拦截器,这里自定义拦截器实现接口IExceptionFilter,接口会要求实现OnException方法,当系统发生未捕获的异常时就会触发这个方法。这里全局异常信息最好能放入数据库里,方便后台查询,再就是抛异常后最好能给负责人发邮件和发送报警短信,也可以直接拨打电话。

public class GlobalExceptionFilter : IExceptionFilter
{private IWebHostEnvironment _env;private ILogger<GlobalExceptionFilter> _logger;public GlobalExceptionFilter(IWebHostEnvironment _env,ILogger<GlobalExceptionFilter> _logger){this._env = _env;this._logger = _logger;}public void OnException(ExceptionContext context){if (context.Exception.GetType() == typeof(BusException)){//如果是自定义异常,则不做处理}else{}//日志入库//向负责人发报警邮件,异步//向负责人发送报警短信或者报警电话,异步Exception ex = context.Exception;//这里给系统分配标识,监控异常肯定不止一个系统。int sysId = 1;//这里获取服务器ip时,需要考虑如果是使用nginx做了负载,这里要兼容负载后的ip,//监控了ip方便定位到底是那台服务器出故障了string ip = context.HttpContext.Connection.RemoteIpAddress.ToString();_logger.LogError($"系统编号:{sysId},主机IP:{ip},堆栈信息:{ex.StackTrace},异常描述:{ex.Message}");context.Result = new JsonResult(ResultBody.error(ex.Message));context.ExceptionHandled = true;}
}

在Startup.ConfigureServices方法里注入全局异常处理拦截器。

public void ConfigureServices(IServiceCollection services)
{services.AddControllersWithViews();//注入全局异常处理services.AddMvc(option =>{option.Filters.Add(typeof(GlobalExceptionFilter));});
}

OK,定义了拦截器后,我们自己抛一个未捕获的异常试试。如图,都会返回统一的JSON返回值。

中间件

定义中间件,定义中间件时先导入日志命名空间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/sword-successful/p/11771858.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

Asp.net Core全局异常监控和记录日志相关推荐

  1. invoke方法是做啥的_使用 NLog 给 Asp.Net Core 做请求监控

    为了减少由于单个请求挂掉而拖垮整站的情况发生,给所有请求做统计是一个不错的解决方法,通过观察哪些请求的耗时比较长,我们就可以找到对应的接口.代码.数据表,做有针对性的优化可以提高效率.在 asp.ne ...

  2. Android UncaughtExceptionHandler 全局异常监控

    2019独角兽企业重金招聘Python工程师标准>>> 一.全局捕获异常 为了解决这样的问题,我们需要能够及时的捕获异常,但要捕获的地方是在太多,因此,我们需要进行全局性的异常捕获, ...

  3. android bug监控,Android UncaughtExceptionHandler 全局异常监控

    一.全局捕获异常 为了解决这样的问题,我们需要能够及时的捕获异常,但要捕获的地方是在太多,因此,我们需要进行全局性的异常捕获,那么如何捕获全局异常呢? 答案是UncaughtExceptionHand ...

  4. ASP.NET Core 网站发布到Linux服务器

    长期以来,使用.NET开发的应用只能运行在Windows平台上面,而目前国内蓬勃发展的互联网公司由于成本的考虑,大量使用免费的Linux平台,这就使得.NET空有一身绝技但无法得到广大的施展空间,.N ...

  5. 如何在 ASP.NET Core 中实现全局异常拦截

    异常是一种运行时错误,当异常没有得到适当的处理,很可能会导致你的程序意外终止,这篇就来讨论一下如何在 ASP.Net Core MVC 中实现全局异常处理,我会用一些 样例代码 和 截图 来说明这些概 ...

  6. 获取异常信息_如何在 ASP.NET Core 中实现全局异常拦截

    异常是一种运行时错误,当异常没有得到适当的处理,很可能会导致你的程序意外终止,这篇就来讨论一下如何在 ASP.Net Core MVC 中实现全局异常处理,我会用一些 样例代码 和 截图 来说明这些概 ...

  7. .NET Core整合log4net以及全局异常捕获实现2

    Startup代码 1 public static ILoggerRepository repository { get; set; } 2 public Startup(IConfiguration ...

  8. asp.net core添加全局异常处理及log4net、Nlog应用

    一.介绍 此篇文章将会介绍项目的全局异常收集以及采用log4net或者NLog记录. 众所周知,一旦自己的项目报错,如果没有进行处理都是显示不友好的,有得甚至直接爆出错误页面,看的也是很奇怪. 为了避 ...

  9. 如何在 ASP.NET Core 中为 gRPC 服务添加全局异常处理 ?

    咨询区 Dmitriy 我在 ASP.NET Core 中使用 GRPC.ASPNETCore 工具包写 gRPC 服务,现在我想实现 gRPC 的异常全局拦截,我的代码如下: app.UseExce ...

最新文章

  1. 湖大深大A级学科数超南开,华科文科胜过武大!泰晤士的首份高校评级结果,让人有点方...
  2. IE6中PNG图片背景无法透明显示的最佳解决方案
  3. RMAN报错:ORA-19573: 无法获得 exclusive 入队
  4. docker加载新的镜像后repository和tag名称都为none的解决方法
  5. 云主机安mysql_如何在云服务器 ECS 安装 MySQL
  6. 27. 二叉搜索树与双向链表(C++版本)
  7. [Windows]GFlag内存泄漏排查
  8. 用户故事与敏捷方法—用户故事的优势
  9. java velocity 语法_Velocity语法
  10. 【数据分析与挖掘】淘宝用户行为分析(带数据集和代码)
  11. JPA ERROR: value too long for type character varying(100)
  12. Espresso测试示例
  13. 计算机三维课设的目的和意义,三维动漫设计开题报告
  14. 英语生疏了,每日至少一句吧
  15. 【刷题日记】网易——牛牛的闹钟
  16. linux扩展堆函数,linux下进程堆栈下溢出判断及扩展实现
  17. 翻译:确认中的处理控制(CO1P)
  18. DELL Inspiron(灵越)14-5648拆机指南
  19. SRCNN与FSRCNN
  20. AI头发笔刷_Photoshop拥有强大的AI工具 可能会让您质疑什么是真实的

热门文章

  1. objective-c中的static
  2. 页面上指定类型的控件的样式添加
  3. 用计算机算算术平方根顺序是ON然后是什么,第2课时用计算器求一个正数的算术平方根.ppt...
  4. 三、Python-列表
  5. 亚信科技数据库AntDB通过金融分布式事务数据库标准测试
  6. 【跃迁之路】【599天】程序员高效学习方法论探索系列(实验阶段356-2018.09.27)...
  7. 2018暑假集训---递推递归----一只小蜜蜂hdu2044
  8. chrome调试工具高级不完整使用指南(基础篇)
  9. Java中数据是如何存储
  10. 简单而不简陋﹣wp7视觉点滴