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
use std::{sync::Arc, time::SystemTime};
use arc_swap::ArcSwapOption;
use crate::{default_logger, Logger, Record};
#[derive(Default)]
pub struct LogCrateProxy {
logger: ArcSwapOption<Logger>,
}
impl LogCrateProxy {
pub(crate) fn new() -> Self {
Self::default()
}
pub fn swap_logger(&self, logger: Option<Arc<Logger>>) -> Option<Arc<Logger>> {
self.logger.swap(logger)
}
pub fn set_logger(&self, logger: Option<Arc<Logger>>) {
self.swap_logger(logger);
}
fn logger(&self) -> Arc<Logger> {
self.logger.load_full().unwrap_or_else(default_logger)
}
}
impl log::Log for LogCrateProxy {
fn enabled(&self, metadata: &log::Metadata) -> bool {
self.logger().should_log(metadata.level().into())
}
fn log(&self, record: &log::Record) {
let logger = self.logger();
let record = Record::from_log_crate_record(&logger, record, SystemTime::now());
logger.log(&record)
}
fn flush(&self) {
self.logger().flush()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::test_utils::*;
#[test]
fn proxy() {
crate::init_log_crate_proxy().unwrap();
log::set_max_level(log::LevelFilter::Debug);
let sink = Arc::new(CounterSink::new());
crate::log_crate_proxy().set_logger(Some(Arc::new(
test_logger_builder().sink(sink.clone()).build(),
)));
assert_eq!(sink.log_count(), 0);
log::info!("hello");
log::error!("world");
assert_eq!(sink.log_count(), 2);
assert_eq!(
sink.payloads(),
vec!["hello".to_string(), "world".to_string()]
);
}
}