Asp.net Core全局異常監控和記錄日誌

来源:https://www.cnblogs.com/sword-successful/archive/2019/10/31/11771858.html
-Advertisement-
Play Games

前言 系統異常監控可以說是重中之重,系統不可能一直運行良好,開發和運維也不可能24小時盯著系統,系統拋異常後我們應當在第一時間收到異常信息。在Asp.net Core里我使用攔截器和中間件兩種方式來監控異常。全局異常監控的數據最好還是寫入資料庫,方便查詢。 配置NLog NLog配置文件 註入NLo ...


前言

          系統異常監控可以說是重中之重,系統不可能一直運行良好,開發和運維也不可能24小時盯著系統,系統拋異常後我們應當在第一時間收到異常信息。在Asp.net Core里我使用攔截器和中間件兩種方式來監控異常。全局異常監控的數據最好還是寫入資料庫,方便查詢。

配置NLog

QQ截圖20191031103620.jpg

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返回值。
QQ截圖20191031101525.jpg
如果未使用全局異常捕獲,則直接拋出如下異常
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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Json.NET常用方法彙總(可解決日常百分之90的需求) 0.Json.NET基礎用法 首先去官網下載最新的Newtonsoft.Json.dll(也可以使用VS自帶的NuGet搜索Json.NET下載(下載下圖第二個))並引用至項目。 (1)序列化實體類(將實體類對象序列化為Json字元串) 結 ...
  • 最近需要用到矩形相交演算法的簡單應用,所以特地拿一個很簡單的演算法出來供新手參考,為什麼說是給新手的參考呢因為這個演算法效率並不是很高,但是這個演算法只有簡簡單單的三行。程式使用了兩種方法來判斷是否重疊/相交,如果有興趣可以看一下,如果覺得有bug可以留言。代碼僅供參考。 C#中矩形的方法為Rectangl ...
  • 一、背景 因編程的基礎差,因此最近開始鞏固學習C#基礎,後期把自己學習的東西,總結相應文章中,有不足處請大家多多指教。 二、簡介 當我們程式出現問題時,我們通過斷點調試來追蹤找到問題點。 1.斷點調試: F10:逐步語句調試 F11:逐過程的調試 2.監視調試 3.即時視窗 三、實例 如圖所示: ...
  • 一、“老生常談”值類型與引用類型 眾所周知,.NET類型系統由 類、結構、枚舉、介面 和 委托 組成。而根據記憶體分配的方式來區分,所有的類型又被分為 值類型 與 引用類型。 一說到值類型,大多數人都會自信地說,“值類型不就是 int,float,double...還有...額...還有啥來著?”。然 ...
  • 一、背景 因編程的基礎差,因此最近開始鞏固學習C#基礎,後期把自己學習的東西,總結相應文章中,有不足處請大家多多指教。 二、簡介 變數:用來在電腦存儲數據 三、語法 第一種: 變數類型 變數名; 變數名=值; 描述:“=”號:在這不是等於的意思,是賦值的意思,表示把等號右邊的值賦值給左邊的變數。 ...
  • 地圖坐標系目前包括: 地球坐標 (WGS84) WGS84:World Geodetic System 1984,是為GPS全球定位系統使用而建立的坐標系統。 國際標準,從 GPS 設備中取出的數據的坐標系 國際地圖提供商使用的坐標系 火星坐標 (GCJ-02)也叫國測局坐標系 GCJ-02是由中國 ...
  • 背景介紹 最近使用WebApi開發一套對外介面,主要是數據的外送以及結果回傳,介面沒什麼難度,採用WebApi+EF的架構簡單創建一個模板工程,使用template生成一套WebApi介面,去掉put、delete等操作,修改一下就可以上線。這些都不在話下,反正網上一大堆教程,隨便找那個step b... ...
  • 一、背景 因編程的基礎差,因此最近開始鞏固學習C#基礎,後期把自己學習的東西,總結相應文章中,有不足處請大家多多指教。 二、簡介 我們在程式中經常會出現各種各樣的異常,你如果想要你的程式更加穩定性,在你的代碼中應該經常使用try-cath來進行異常捕獲。 哪行代碼有可能出現異常,我們就使用try-c ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...