use crate::{gstr, log as glib_log, log_structured_array, translate::*, LogField};
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum GlibLoggerFormat {
Plain,
LineAndFile,
Structured,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum GlibLoggerDomain {
None,
CrateTarget,
CratePath,
}
#[derive(Debug)]
pub struct GlibLogger {
format: GlibLoggerFormat,
domain: GlibLoggerDomain,
}
impl GlibLogger {
pub const fn new(format: GlibLoggerFormat, domain: GlibLoggerDomain) -> Self {
Self { format, domain }
}
fn level_to_glib(level: rs_log::Level) -> crate::LogLevel {
match level {
rs_log::Level::Error => crate::LogLevel::Critical,
rs_log::Level::Warn => crate::LogLevel::Warning,
rs_log::Level::Info => crate::LogLevel::Info,
rs_log::Level::Debug => crate::LogLevel::Debug,
rs_log::Level::Trace => crate::LogLevel::Debug,
}
}
#[doc(alias = "g_log")]
fn write_log(domain: Option<&str>, level: rs_log::Level, message: &std::fmt::Arguments<'_>) {
unsafe {
use std::fmt::Write;
let mut message_builder = crate::GStringBuilder::default();
if write!(&mut message_builder, "{}", message).is_err() {
return;
}
let message = message_builder.into_string();
crate::ffi::g_log(
domain.to_glib_none().0,
GlibLogger::level_to_glib(level).into_glib(),
b"%s\0".as_ptr() as *const _,
ToGlibPtr::<*const std::os::raw::c_char>::to_glib_none(&message).0,
);
}
}
fn write_log_structured(
domain: Option<&str>,
level: rs_log::Level,
file: Option<&str>,
line: Option<u32>,
func: Option<&str>,
message: &str,
) {
let mut line_buffer = [0u8; 16];
let line = {
use std::io::{Cursor, Write};
let mut c = Cursor::new(line_buffer.as_mut_slice());
match line {
Some(lineno) => write!(&mut c, "{lineno}").ok(),
None => write!(&mut c, "<unknown line>").ok(),
};
let pos = c.position() as usize;
&line_buffer[..pos]
};
let glib_level = GlibLogger::level_to_glib(level);
let fields = [
LogField::new(gstr!("PRIORITY"), glib_level.priority().as_bytes()),
LogField::new(
gstr!("CODE_FILE"),
file.unwrap_or("<unknown file>").as_bytes(),
),
LogField::new(gstr!("CODE_LINE"), line),
LogField::new(
gstr!("CODE_FUNC"),
func.unwrap_or("<unknown module path>").as_bytes(),
),
LogField::new(gstr!("MESSAGE"), message.as_bytes()),
LogField::new(gstr!("GLIB_DOMAIN"), domain.unwrap_or("default").as_bytes()),
];
log_structured_array(glib_level, &fields);
}
}
impl rs_log::Log for GlibLogger {
fn enabled(&self, _: &rs_log::Metadata) -> bool {
true
}
fn log(&self, record: &rs_log::Record) {
if !self.enabled(record.metadata()) {
return;
}
let domain = match &self.domain {
GlibLoggerDomain::None => None,
GlibLoggerDomain::CrateTarget => Some(record.metadata().target()),
GlibLoggerDomain::CratePath => record.module_path(),
};
match self.format {
GlibLoggerFormat::Plain => {
GlibLogger::write_log(domain, record.level(), record.args());
}
GlibLoggerFormat::LineAndFile => {
match (record.file(), record.line()) {
(Some(file), Some(line)) => {
GlibLogger::write_log(
domain,
record.level(),
&format_args!("{}:{}: {}", file, line, record.args()),
);
}
(Some(file), None) => {
GlibLogger::write_log(
domain,
record.level(),
&format_args!("{}: {}", file, record.args()),
);
}
_ => {
GlibLogger::write_log(domain, record.level(), record.args());
}
};
}
GlibLoggerFormat::Structured => {
let args = record.args();
let message = if let Some(s) = args.as_str() {
s
} else {
&args.to_string()
};
GlibLogger::write_log_structured(
domain,
record.level(),
record.file(),
record.line(),
record.module_path(),
message,
);
}
};
}
fn flush(&self) {}
}
pub fn rust_log_handler(domain: Option<&str>, level: glib_log::LogLevel, message: &str) {
let level = match level {
glib_log::LogLevel::Error | glib_log::LogLevel::Critical => rs_log::Level::Error,
glib_log::LogLevel::Warning => rs_log::Level::Warn,
glib_log::LogLevel::Message | glib_log::LogLevel::Info => rs_log::Level::Info,
glib_log::LogLevel::Debug => rs_log::Level::Debug,
};
rs_log::log!(target: domain.unwrap_or("<null>"), level, "{}", message);
}
#[macro_export]
#[cfg(any(docsrs, feature = "log_macros"))]
#[cfg_attr(docsrs, doc(cfg(feature = "log_macros")))]
macro_rules! error {
(target: $target:expr, $($arg:tt)+) => (
$crate::rs_log::log!(target: $target, $crate::rs_log::Level::Error, $($arg)+);
);
($($arg:tt)+) => (
$crate::rs_log::log!(target: G_LOG_DOMAIN, $crate::rs_log::Level::Error, $($arg)+);
)
}
#[macro_export]
#[cfg(any(docsrs, feature = "log_macros"))]
#[cfg_attr(docsrs, doc(cfg(feature = "log_macros")))]
macro_rules! warn {
(target: $target:expr, $($arg:tt)+) => (
$crate::rs_log::log!(target: $target, $crate::rs_log::Level::Warn, $($arg)+);
);
($($arg:tt)+) => (
$crate::rs_log::log!(target: G_LOG_DOMAIN, $crate::rs_log::Level::Warn, $($arg)+);
)
}
#[macro_export]
#[cfg(any(docsrs, feature = "log_macros"))]
#[cfg_attr(docsrs, doc(cfg(feature = "log_macros")))]
macro_rules! info {
(target: $target:expr, $($arg:tt)+) => (
$crate::rs_log::log!(target: $target, $crate::rs_log::Level::Info, $($arg)+);
);
($($arg:tt)+) => (
$crate::rs_log::log!(target: G_LOG_DOMAIN, $crate::rs_log::Level::Info, $($arg)+);
)
}
#[macro_export]
#[cfg(any(docsrs, feature = "log_macros"))]
#[cfg_attr(docsrs, doc(cfg(feature = "log_macros")))]
macro_rules! debug {
(target: $target:expr, $($arg:tt)+) => (
$crate::rs_log::log!(target: $target, $crate::rs_log::Level::Debug, $($arg)+);
);
($($arg:tt)+) => (
$crate::rs_log::log!(target: G_LOG_DOMAIN, $crate::rs_log::Level::Debug, $($arg)+);
)
}
#[macro_export]
#[cfg(any(docsrs, feature = "log_macros"))]
#[cfg_attr(docsrs, doc(cfg(feature = "log_macros")))]
macro_rules! trace {
(target: $target:expr, $($arg:tt)+) => (
$crate::rs_log::log!(target: $target, $crate::rs_log::Level::Trace, $($arg)+);
);
($($arg:tt)+) => (
$crate::rs_log::log!(target: G_LOG_DOMAIN, $crate::rs_log::Level::Trace, $($arg)+);
)
}