欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Asp.net Core全局异常监控和记录日志

程序员文章站 2022-06-20 17:05:50
前言 系统异常监控可以说是重中之重,系统不可能一直运行良好,开发和运维也不可能24小时盯着系统,系统抛异常后我们应当在第一时间收到异常信息。在Asp.net Core里我使用拦截器和中间件两种方式来监控异常。全局异常监控的数据最好还是写入数据库,方便查询。 配置NLog NLog配置文件 注入NLo ......

前言

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

配置nlog

Asp.net Core全局异常监控和记录日志

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返回值。
Asp.net Core全局异常监控和记录日志
如果未使用全局异常捕获,则直接抛出如下异常
Asp.net Core全局异常监控和记录日志
         客户端抛出异常后,可查看磁盘写入日志,这里看到我关注的系统编号,主机ip,堆栈信息和异常描述信息。
Asp.net Core全局异常监控和记录日志

中间件

定义中间件,定义中间件时先导入日志命名空间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对象更直观些。

参考链接