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
|
use std::sync::LazyLock;
use bevy_log::tracing_subscriber::{
self, EnvFilter, Layer, Registry,
filter::Filtered,
fmt,
layer::{Context, SubscriberExt},
reload::{self, Handle},
util::SubscriberInitExt,
};
use parking_lot::{Mutex, MutexGuard};
use tracing::{Event, Level, Subscriber};
static LOCK: LazyLock<Mutex<()>> = LazyLock::new(|| Mutex::new(()));
pub fn init<'a>() -> MutexGuard<'a, ()> {
init_with_level(Level::WARN)
}
// can't treat these as one layer due to https://github.com/tokio-rs/tracing/issues/1629
struct LayerReloadHandles {
filter: Handle<Filtered<fmt::Layer<Registry>, EnvFilter, Registry>, Registry>,
test: Handle<TestTracingLayer, Registry>,
}
static RELOAD_HANDLES: LazyLock<LayerReloadHandles> = LazyLock::new(|| {
let (filter_layer, filter_reload_handle) = reload::Layer::new(
fmt::layer().with_filter(
EnvFilter::builder()
.with_default_directive(Level::WARN.into())
.from_env_lossy(),
),
);
let (test_layer, test_reload_handle) = reload::Layer::new(TestTracingLayer {
panic_on_level: Level::WARN,
});
tracing_subscriber::registry()
.with(filter_layer.and_then(test_layer))
.init();
LayerReloadHandles {
filter: filter_reload_handle,
test: test_reload_handle,
}
});
pub fn init_with_level<'a>(max_level: Level) -> MutexGuard<'a, ()> {
let lock = LOCK.lock();
RELOAD_HANDLES
.filter
.modify(|layer| {
*layer.filter_mut() = EnvFilter::builder()
.with_default_directive(max_level.into())
.from_env_lossy()
})
.unwrap();
RELOAD_HANDLES
.test
.modify(|layer| layer.panic_on_level = max_level)
.unwrap();
lock
}
struct TestTracingLayer {
panic_on_level: Level,
}
impl<S: Subscriber> Layer<S> for TestTracingLayer {
fn on_event(&self, event: &Event<'_>, _ctx: Context<'_, S>) {
let level = *event.metadata().level();
if level <= self.panic_on_level {
panic!("Logged on level {level}.");
}
}
}
|