use std::time::SystemTime;
use crate::{default_logger, sync::*, utils, LogCrateRecord, Logger};
#[derive(Default)]
pub struct LogCrateProxy {
logger: ArcSwapOption<Logger>,
filter: ArcSwapOption<env_filter::Filter>,
}
impl LogCrateProxy {
#[must_use]
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);
}
pub fn set_filter(&self, filter: Option<env_filter::Filter>) {
self.filter.swap(filter.map(Arc::new));
}
#[must_use]
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 {
let filter = self.filter.load();
utils::is_none_or(filter.as_deref(), |filter| filter.enabled(metadata))
&& self.logger().should_log(metadata.level().into())
}
fn log(&self, record: &log::Record) {
if utils::is_none_or(self.filter.load().as_deref(), |filter| {
filter.matches(record)
}) {
let logger = self.logger();
let record = LogCrateRecord::new(&logger, record, SystemTime::now());
logger.log(&record.as_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(TestSink::new());
crate::log_crate_proxy()
.set_logger(Some(Arc::new(build_test_logger(|b| b.sink(sink.clone())))));
crate::log_crate_proxy().set_filter(Some(
env_filter::Builder::new()
.filter_module(
"spdlog::log_crate_proxy::tests::should_be_filtered_out",
log::LevelFilter::Off,
)
.filter(None, log::LevelFilter::Trace)
.build(),
));
assert_eq!(sink.log_count(), 0);
log::info!("hello");
log::error!("world");
mod should_be_filtered_out {
pub fn log_something() {
log::warn!("this should be filtered out");
}
}
should_be_filtered_out::log_something();
assert_eq!(sink.log_count(), 2);
assert_eq!(
sink.payloads(),
vec!["hello".to_string(), "world".to_string()]
);
}
}