cargo/
macros.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
use std::fmt;

macro_rules! compact_debug {
    (
        impl fmt::Debug for $ty:ident {
            fn fmt(&$this:ident, f: &mut fmt::Formatter) -> fmt::Result {
                let (default, default_name) = $e:expr;
                [debug_the_fields($($field:ident)*)]
            }
        }
    ) => (

        impl fmt::Debug for $ty {
            fn fmt(&$this, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                // Try printing a pretty version where we collapse as many fields as
                // possible, indicating that they're equivalent to a function call
                // that's hopefully enough to indicate what each value is without
                // actually dumping everything so verbosely.
                let mut s = f.debug_struct(stringify!($ty));
                let (default, default_name) = $e;
                let mut any_default = false;

                // Exhaustively match so when fields are added we get a compile
                // failure
                let $ty { $($field),* } = $this;
                $(
                    if *$field == default.$field {
                        any_default = true;
                    } else {
                        s.field(stringify!($field), $field);
                    }
                )*

                if any_default {
                    s.field("..", &crate::macros::DisplayAsDebug(default_name));
                }
                s.finish()
            }
        }
    )
}

pub struct DisplayAsDebug<T>(pub T);

impl<T: fmt::Display> fmt::Debug for DisplayAsDebug<T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        fmt::Display::fmt(&self.0, f)
    }
}