#logging #privacy #arti #tor #debugging

safelog

Conditionally suppress confidential information from logs

20 releases

new 0.4.3 Jan 7, 2025
0.4.2 Dec 2, 2024
0.4.1 Oct 31, 2024
0.3.7 Jun 5, 2024
0.1.1 Jun 24, 2022

#218 in Rust patterns

Download history 1540/week @ 2024-09-19 2316/week @ 2024-09-26 1499/week @ 2024-10-03 1284/week @ 2024-10-10 1560/week @ 2024-10-17 1798/week @ 2024-10-24 2922/week @ 2024-10-31 1564/week @ 2024-11-07 2568/week @ 2024-11-14 2137/week @ 2024-11-21 2822/week @ 2024-11-28 2681/week @ 2024-12-05 2620/week @ 2024-12-12 1495/week @ 2024-12-19 857/week @ 2024-12-26 1710/week @ 2025-01-02

7,282 downloads per month
Used in 65 crates (20 directly)

MIT/Apache

36KB
634 lines

safelog

Mark data as sensitive for logging purposes.

Some information is too sensitive to routinely write to system logs, but must nonetheless sometimes be displayed. This crate provides a way to mark such information, and log it conditionally, but not by default.

Examples

There are two main ways to mark a piece of data as sensitive: by storing it within a Sensitive object long-term, or by wrapping it in a Sensitive object right before passing it to a formatter:

use safelog::{Sensitive, sensitive};

// With this declaration, a student's name and gpa will be suppressed by default
// when passing the student to Debug.
#[derive(Debug)]
struct Student {
   name: Sensitive<String>,
   grade: u8,
   homeroom: String,
   gpa: Sensitive<f32>,
}

// In this function, a user's IP will not be printed by default.
fn record_login(username: &str, ip: &std::net::IpAddr) {
    println!("Login from {} at {}", username, sensitive(ip));
}

You can disable safe-logging globally (across all threads) or locally (across a single thread).

# let debug_mode = true;
# let log_encrypted_data = |_|();
# let big_secret = ();
use safelog::{disable_safe_logging, with_safe_logging_suppressed};

// If we're running in debug mode, turn off safe logging
// globally.  Safe logging will remain disabled until the
// guard object is dropped.
let guard = if debug_mode {
   // This call can fail if safe logging has already been enforced.
   disable_safe_logging().ok()
} else {
   None
};

// If we know that it's safe to record sensitive data with a given API,
// we can disable safe logging temporarily. This affects only the current thread.
with_safe_logging_suppressed(|| log_encrypted_data(big_secret));

An example deployment

This crate was originally created for use in the arti project, which tries to implements the Tor anonymity protocol in Rust. In arti, we want to avoid logging information by default if it could compromise users' anonymity, or create an incentive for attacking users and relays in order to access their logs.

In general, Arti treats the following information as Sensitive:

  • Client addresses.
  • The destinations (target addresses) of client requests.

Arti does not label all private information as Sensitive: when information isn't ever suitable for logging, we omit it entirely.

License: MIT OR Apache-2.0

Dependencies

~0.8–1.5MB
~32K SLoC