mirror of
https://github.com/Zedfrigg/ironbar.git
synced 2025-08-17 06:41:03 +02:00
126 lines
4 KiB
Rust
126 lines
4 KiB
Rust
use color_eyre::Result;
|
|
use dirs::data_dir;
|
|
use glib::{LogLevel, LogWriterOutput};
|
|
use std::{env, panic};
|
|
use strip_ansi_escapes::Writer;
|
|
use tracing::{debug, error, info, warn};
|
|
use tracing_appender::non_blocking::{NonBlocking, WorkerGuard};
|
|
use tracing_appender::rolling::Rotation;
|
|
use tracing_error::ErrorLayer;
|
|
use tracing_subscriber::fmt::{Layer, MakeWriter};
|
|
use tracing_subscriber::prelude::*;
|
|
use tracing_subscriber::{EnvFilter, fmt};
|
|
|
|
struct MakeFileWriter {
|
|
file_writer: NonBlocking,
|
|
}
|
|
|
|
impl MakeFileWriter {
|
|
const fn new(file_writer: NonBlocking) -> Self {
|
|
Self { file_writer }
|
|
}
|
|
}
|
|
|
|
impl<'a> MakeWriter<'a> for MakeFileWriter {
|
|
type Writer = Writer<NonBlocking>;
|
|
|
|
fn make_writer(&'a self) -> Self::Writer {
|
|
Writer::new(self.file_writer.clone())
|
|
}
|
|
}
|
|
|
|
pub fn install_logging() -> Result<WorkerGuard> {
|
|
// Disable backtraces by default
|
|
if env::var("RUST_LIB_BACKTRACE").is_err() {
|
|
// as this is the very first thing we do (before runtimes are set up)
|
|
// we can be sure that it only runs in a single-thread context
|
|
unsafe {
|
|
env::set_var("RUST_LIB_BACKTRACE", "0");
|
|
}
|
|
}
|
|
|
|
// keep guard in scope
|
|
// otherwise file logging drops
|
|
let guard = install_tracing()?;
|
|
|
|
let hook_builder = color_eyre::config::HookBuilder::default();
|
|
let (panic_hook, eyre_hook) = hook_builder.into_hooks();
|
|
|
|
eyre_hook.install()?;
|
|
|
|
// custom hook allows tracing_appender to capture panics
|
|
panic::set_hook(Box::new(move |panic_info| {
|
|
error!("{}", panic_hook.panic_report(panic_info));
|
|
}));
|
|
|
|
Ok(guard)
|
|
}
|
|
|
|
/// Installs tracing into the current application.
|
|
///
|
|
/// The returned `WorkerGuard` must remain in scope
|
|
/// for the lifetime of the application for logging to file to work.
|
|
fn install_tracing() -> Result<WorkerGuard> {
|
|
const DEFAULT_LOG: &str = "info";
|
|
const DEFAULT_FILE_LOG: &str = "warn";
|
|
|
|
let fmt_layer = fmt::layer().with_target(true).with_line_number(true);
|
|
let filter_layer =
|
|
EnvFilter::try_from_env("IRONBAR_LOG").or_else(|_| EnvFilter::try_new(DEFAULT_LOG))?;
|
|
|
|
let file_filter_layer = EnvFilter::try_from_env("IRONBAR_FILE_LOG")
|
|
.or_else(|_| EnvFilter::try_new(DEFAULT_FILE_LOG))?;
|
|
|
|
let log_path = data_dir().unwrap_or(env::current_dir()?).join("ironbar");
|
|
|
|
let appender = tracing_appender::rolling::Builder::new()
|
|
.rotation(Rotation::DAILY)
|
|
.filename_prefix("ironbar")
|
|
.filename_suffix("log")
|
|
.max_log_files(3)
|
|
.build(log_path)?;
|
|
|
|
let (file_writer, guard) = tracing_appender::non_blocking(appender);
|
|
|
|
tracing_subscriber::registry()
|
|
.with(filter_layer)
|
|
.with(fmt_layer)
|
|
.with(ErrorLayer::default())
|
|
.with(
|
|
Layer::default()
|
|
.with_writer(MakeFileWriter::new(file_writer))
|
|
.with_ansi(false)
|
|
.with_filter(file_filter_layer),
|
|
)
|
|
.init();
|
|
|
|
glib::log_set_writer_func(|level, fields| {
|
|
const KEY_DOMAIN: &str = "GLIB_DOMAIN";
|
|
const KEY_MESSAGE: &str = "MESSAGE";
|
|
|
|
let domain = fields
|
|
.iter()
|
|
.find(|f| f.key() == KEY_DOMAIN)
|
|
.and_then(|f| f.value_str())
|
|
.unwrap_or("Glib Unknown");
|
|
|
|
let message = fields
|
|
.iter()
|
|
.find(|f| f.key() == KEY_MESSAGE)
|
|
.and_then(|f| f.value_str())
|
|
.unwrap_or_default();
|
|
|
|
match level {
|
|
LogLevel::Error => error!(target: "GTK", "[{domain}] {message}"),
|
|
LogLevel::Critical => error!(target: "GTK", "[{domain}] CRITICAL: {message}"),
|
|
LogLevel::Warning => warn!(target: "GTK", "[{domain}] {message}"),
|
|
LogLevel::Message => info!(target: "GTK", "[{domain}] MESSAGE: {message}"),
|
|
LogLevel::Info => info!(target: "GTK", "[{domain}] MESSAGE: {message}"),
|
|
LogLevel::Debug => debug!(target: "GTK", "[{domain}] {message}"),
|
|
}
|
|
|
|
LogWriterOutput::Handled
|
|
});
|
|
|
|
Ok(guard)
|
|
}
|