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
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
use std::sync::OnceLock;

use tracing::{Dispatch, Event, Metadata};
use tracing::subscriber::{Subscriber, Interest};
use tracing::span::{Attributes, Id, Record};

use tracing_subscriber::reload;
use tracing_subscriber::registry::{Registry, LookupSpan};
use tracing_subscriber::layer::{Context, Layer, Layered, SubscriberExt};
use tracing_subscriber::util::SubscriberInitExt;

use crate::config::Config;
use crate::trace::subscriber::{Compact, Pretty, RequestId, RequestIdLayer, RocketFmt};
use crate::trace::TraceFormat;

/// A subscriber that is either a [`Pretty`] or [`Compact`] [`RocketFmt`].
pub struct RocketDynFmt {
    inner: either::Either<RocketFmt<Compact>, RocketFmt<Pretty>>,
}

impl From<RocketFmt<Compact>> for RocketDynFmt {
    fn from(value: RocketFmt<Compact>) -> Self {
        RocketDynFmt { inner: either::Either::Left(value) }
    }
}

impl From<RocketFmt<Pretty>> for RocketDynFmt {
    fn from(value: RocketFmt<Pretty>) -> Self {
        RocketDynFmt { inner: either::Either::Right(value) }
    }
}

impl RocketDynFmt {
    /// Creates a new `RocketDynFmt` subscriber given a `Config`.
    ///
    /// [`Config::log_format`] determines which `RocketFmt` subscriber (either
    /// [`Pretty`] or [`Compact`]) is used.
    ///
    /// If `config` is `None`, [`Config::debug_default()`] is used, which uses
    /// the [`Pretty`] subscriber by default.
    pub fn new(config: Option<&Config>) -> Self {
        let default = Config::debug_default();
        let workers = config.map_or(default.workers, |c| c.workers);
        let colors = config.map_or(default.cli_colors, |c| c.cli_colors);
        let level = config.map_or(default.log_level, |c| c.log_level);
        let format = config.map_or(default.log_format, |c| c.log_format);

        match format {
            TraceFormat::Pretty => Self::from(RocketFmt::<Pretty>::new(workers, colors, level)),
            TraceFormat::Compact => Self::from(RocketFmt::<Compact>::new(workers, colors, level)),
        }
    }

    pub(crate) fn init(config: Option<&Config>) {
        type Handle = reload::Handle<RocketDynFmt, Layered<RequestIdLayer, Registry>>;

        static HANDLE: OnceLock<Handle> = OnceLock::new();

        // Do nothing if there's no config and we've already initialized.
        if config.is_none() && HANDLE.get().is_some() {
            return;
        }

        let formatter = Self::new(config);
        if let Some(handle) = HANDLE.get() {
            return assert!(handle.modify(|layer| *layer = formatter).is_ok());
        }

        let (layer, reload_handle) = reload::Layer::new(formatter);
        let result = tracing_subscriber::registry()
            .with(RequestId::layer())
            .with(layer)
            .try_init();

        if result.is_ok() {
            assert!(HANDLE.set(reload_handle).is_ok());
        }
    }
}

macro_rules! forward {
    ($T:ident => $(& $r:tt)? $method:ident ( $($p:ident : $t:ty),* ) $(-> $R:ty)?) => {
        #[inline(always)]
        fn $method(& $($r)? self $(, $p : $t)*) $(-> $R)? {
            match & $($r)* self.inner {
                either::Either::Left(layer) => Layer::<$T>::$method(layer, $($p),*),
                either::Either::Right(layer) => Layer::<$T>::$method(layer, $($p),*),
            }
        }
    };
}

impl<S: Subscriber + for<'a> LookupSpan<'a>> Layer<S> for RocketDynFmt {
    forward!(S => on_register_dispatch(subscriber: &Dispatch));
    forward!(S => &mut on_layer(subscriber: &mut S));
    forward!(S => register_callsite(metadata: &'static Metadata<'static>) -> Interest);
    forward!(S => enabled(metadata: &Metadata<'_>, ctx: Context<'_, S>) -> bool);
    forward!(S => on_new_span(attrs: &Attributes<'_>, id: &Id, ctx: Context<'_, S>));
    forward!(S => on_record(_span: &Id, _values: &Record<'_>, _ctx: Context<'_, S>));
    forward!(S => on_follows_from(_span: &Id, _follows: &Id, _ctx: Context<'_, S>));
    forward!(S => event_enabled(_event: &Event<'_>, _ctx: Context<'_, S>) -> bool);
    forward!(S => on_event(_event: &Event<'_>, _ctx: Context<'_, S>));
    forward!(S => on_enter(_id: &Id, _ctx: Context<'_, S>));
    forward!(S => on_exit(_id: &Id, _ctx: Context<'_, S>));
    forward!(S => on_close(_id: Id, _ctx: Context<'_, S>));
    forward!(S => on_id_change(_old: &Id, _new: &Id, _ctx: Context<'_, S>));
}