練習:泛型 Logger

我們來設計一個簡單的記錄公用程式,使用 Logger 特徵搭配 log 方法。如果程式碼可能會記錄相關進度,就可以採用 &impl Logger。在測試過程中,這可能會將訊息置於測試記錄檔中;而在實際版本中,則會將訊息傳送至記錄伺服器。

不過,下方的 StderrLogger 會記錄詳細程度不限的所有訊息。您的任務是編寫 VerbosityFilter 型別,忽略超出詳細程度上限的訊息。

以下是常見模式:結構體包裝一個特徵實作項目,並實作該相同特徵,在程序中加入行為。想一想,還有哪些其他類型的包裝函式可能在記錄公用程式中派上用場?

use std::fmt::Display;

pub trait Logger {
    /// Log a message at the given verbosity level.
    fn log(&self, verbosity: u8, message: impl Display);
}

struct StderrLogger;

impl Logger for StderrLogger {
    fn log(&self, verbosity: u8, message: impl Display) {
        eprintln!("verbosity={verbosity}: {message}");
    }
}

fn do_things(logger: &impl Logger) {
    logger.log(5, "FYI");
    logger.log(2, "Uhoh");
}

// TODO: Define and implement `VerbosityFilter`.

fn main() {
    let l = VerbosityFilter { max_verbosity: 3, inner: StderrLogger };
    do_things(&l);
}