Exercise: Logger Trait

Let’s design a simple logging utility, using a trait Logger with a log method. Code which might log its progress can then take an &impl Logger. In testing, this might put messages in the test logfile, while in a production build it would send messages to a log server.

However, the StdoutLogger given below logs all messages, regardless of verbosity. Your task is to write a VerbosityFilter type that will ignore messages above a maximum verbosity.

This is a common pattern: a struct wrapping a trait implementation and implementing that same trait, adding behavior in the process. What other kinds of wrappers might be useful in a logging utility?

  1. pub trait Logger {
  2.     /// Log a message at the given verbosity level.
  3.     fn log(&self, verbosity: u8, message: &str);
  4. }
  5. struct StdoutLogger;
  6. impl Logger for StdoutLogger {
  7.     fn log(&self, verbosity: u8, message: &str) {
  8.         println!("verbosity={verbosity}: {message}");
  9.     }
  10. }
  11. // TODO: Define and implement `VerbosityFilter`.
  12. fn main() {
  13.     let logger = VerbosityFilter { max_verbosity: 3, inner: StdoutLogger };
  14.     logger.log(5, "FYI");
  15.     logger.log(2, "Uhoh");
  16. }