Rust 實現日誌記錄功能

二次元攻城狮發表於2024-04-01

目錄
  • log 日誌庫標準
    • 簡單示例
    • 使用方法
      • 庫的開發者
      • 應用開發者
      • 日誌庫開發者
  • 使用 log4rs
    • 新增依賴
    • 配置檔案
    • 執行專案
  • 參考文章

log 日誌庫標準

log 是 Rust 的日誌門面庫,由官方積極維護可以放心使用。它是Rust的日誌門面,相應的日誌 API 已成為事實上的標準被其它日誌框架所使用,有了日誌門面開發者可以很方便切換自己的日誌框架。

簡單示例

建立一個名為 log_test 二進位制專案:

cargo new log_test

執行以下命令,引入 log 依賴:

cargo add log

修改 main.rs 的程式碼如下:

use log::{info, warn,error,trace};

fn main() {
    // 記錄日誌
    info!("This is an info message");
    warn!("This is a warning message");
    error!("This is an error message");
    trace!("This is an trace message");
}

執行上面的示例,實際上看不到任何輸出。因為 log 僅僅是日誌門面庫,它並不具備完整的日誌庫功能

使用方法

因為 log 僅僅是日誌門面庫,對於庫和應用的開發者來說使用方法是有區別的,這也是 log 包這麼設計的優勢所在。

庫的開發者

作為庫的開發者,你只要在庫中使用門面庫即可,將具體的日誌庫交給使用者去選擇和繫結:

use log::{info, trace, warn};
pub fn deal_with_something() {
    // 開始處理

    // 記錄一些日誌
    trace!("a trace log");
    info!("a info long: {}", "abc");
    warn!("a warning log: {}, retrying", err);

    // 結束處理
}

應用開發者

如果是應用開發者,就需要去選擇一個具體的日誌庫了。

目前已經有了不少日誌庫實現,官方在 github 上也推薦了一些 ,主要分為以下幾類:

  • 簡單的最小記錄器,如 env_logger 等
  • 複雜的可配置框架,如 log4rs 等
  • 其他設施的介面卡,如 syslog、db_logger 等
  • 對於 WebAssembly 二進位制檔案:console_log
  • 對於動態庫:需要在日誌上構造一個 FFI 安全包裝器,以便在庫中進行初始化。
  • 實用程式,如 alterable_logger 等

log 還提供了 set_logger 函式用於設定日誌庫,set_max_level 用於設定最大日誌級別。但是選用的具體日誌庫往往會提供更高階的 API,無需手動呼叫這兩個函式。

日誌庫開發者

對於日誌庫開發者而言,自然要實現自己的 Log 特徵:

use log::{Record, Level, Metadata};
struct SimpleLogger;
impl log::Log for SimpleLogger {
    fn enabled(&self, metadata: &Metadata) -> bool {
        metadata.level() <= Level::Info
    }
    fn log(&self, record: &Record) {
        if self.enabled(record.metadata()) {
            println!("{} - {}", record.level(), record.args());
        }
    }
    fn flush(&self) {}
}

除此之外,還需要包裝下 set_logger 和 set_max_level:

use log::{SetLoggerError, LevelFilter};
static LOGGER: SimpleLogger = SimpleLogger;
pub fn init() -> Result<(), SetLoggerError> {
    log::set_logger(&LOGGER)
        .map(|()| log::set_max_level(LevelFilter::Info))
}

然後再main函式里面設定全域性記錄器:

use log::{info, warn,error,trace};

fn main() {
    //設定日誌
    init();
    // 記錄日誌
    info!("This is an info message");
    warn!("This is a warning message");
    error!("This is an error message");
    trace!("This is an trace message");
}

執行後終端輸出如下,因為設定日誌等級為Info,所以沒有輸出Trace等級日誌:

INFO - This is an info message
WARN - This is a warning message
ERROR - This is an error message

使用 log4rs

log4rs 是一個高度可配置的日誌框架,以 Java 的 Logback 和 log4j 庫為模型。

新增依賴

為專案新增 log4rs 依賴:

cargo add log4rs

配置檔案

在專案根目錄下,建立一個 log4rs.yaml 配置檔案,並新增以下內容:

refresh_rate: 30 seconds

appenders:
  stdout:
    kind: console
    encoder:
      pattern: "{d(%Y-%m-%d %H:%M:%S.%f)} [{l}] {t} - {m}{n}"
  
  rolling_file:
    kind: rolling_file
    path: logs/test.log
    append: true 
    encoder:
      pattern: "{d(%Y-%m-%d %H:%M:%S.%f)} [{l}] {t} - {m}{n}"
    policy:
      kind: compound
      trigger:
        kind: size
        limit: 10 mb
      roller:        
        kind: fixed_window
        pattern: logs/test.{}.log
        base: 1
        count: 5
        
root:
  level: info 
  appenders:
    - stdout
    - rolling_file

上面配置檔案設定日誌輸出到控制檯、檔案,檔案按10 mb大小滾動,只保留最近五個檔案。各個配置欄位的具體含義可以參考配置

  • refresh_rate:用於確定 log4rs 掃描配置檔案以查詢更改的頻率,如果發現更改,記錄器將自動重新配置
  • appender: 負責將日誌收集到檔案、控制檯或系統日誌, 可配置多個
  • stdout、rolling_file:追加器的唯一標識字串,自己隨便定義,它的 kind 欄位只支援consolefilerolling_file 三種實現
  • encoder: 負責將 log 資訊轉換為合適的格式, 如固定格式的平文字或json
  • pattern:編碼模板,格式可配置,具體格式詳見pattern
  • policy:策略欄位,策略必須具有 kind 欄位,預設(且僅受支援)策略為 kind: compound
  • trigger:觸發器欄位用於指示何時滾動日誌檔案,支援 sizetime 兩種型別,這裡使用的是按大小

執行專案

修改main.rs內容如下:

use log::*;
use log4rs;

fn main() {
    log4rs::init_file("log4rs.yaml", Default::default()).unwrap();

    for i in 1..=1000 {
        info!("This is loop iteration {}", i);
    }
}

執行結果:

2024-04-01 15:43:28.596832500 [INFO] hello_world - This is loop iteration 1

參考文章

  • 日誌門面 log
  • Crate log
  • rust輸出日誌檔案
  • [Rust Crate] log4rs Rust log 庫

相關文章