mod format;
mod json_logger;
mod stdout_logger;
use log::{Level, LevelFilter, Log, Metadata, Record};
use serde::{Deserialize, Serialize};
use std::fmt;
use self::{json_logger::JsonLogger, stdout_logger::StdoutLogger};
use crate::Frame;
const DEFMT_TARGET_MARKER: &str = "defmt@";
pub fn log_defmt(
frame: &Frame<'_>,
file: Option<&str>,
line: Option<u32>,
module_path: Option<&str>,
) {
let timestamp = frame
.display_timestamp()
.map(|ts| ts.to_string())
.unwrap_or_default();
let level = frame.level().map(|level| match level {
crate::Level::Trace => Level::Trace,
crate::Level::Debug => Level::Debug,
crate::Level::Info => Level::Info,
crate::Level::Warn => Level::Warn,
crate::Level::Error => Level::Error,
});
let target = format!(
"{}{}",
DEFMT_TARGET_MARKER,
serde_json::to_value(Payload { timestamp, level }).unwrap()
);
log::logger().log(
&Record::builder()
.args(format_args!("{}", frame.display_message()))
.target(&target)
.module_path(module_path)
.file(file)
.line(line)
.build(),
);
}
pub fn is_defmt_frame(metadata: &Metadata) -> bool {
metadata.target().starts_with(DEFMT_TARGET_MARKER)
}
pub struct DefmtRecord<'a> {
log_record: &'a Record<'a>,
payload: Payload,
}
#[derive(Deserialize, Serialize)]
struct Payload {
level: Option<Level>,
timestamp: String,
}
impl<'a> DefmtRecord<'a> {
pub fn new(log_record: &'a Record<'a>) -> Option<Self> {
let target = log_record.metadata().target();
target
.strip_prefix(DEFMT_TARGET_MARKER)
.map(|payload| Self {
log_record,
payload: serde_json::from_str(payload).expect("malformed 'payload'"),
})
}
pub fn timestamp(&self) -> &str {
self.payload.timestamp.as_str()
}
pub fn level(&self) -> Option<Level> {
self.payload.level
}
pub fn args(&self) -> &fmt::Arguments<'a> {
self.log_record.args()
}
pub fn module_path(&self) -> Option<&'a str> {
self.log_record.module_path()
}
pub fn file(&self) -> Option<&'a str> {
self.log_record.file()
}
pub fn line(&self) -> Option<u32> {
self.log_record.line()
}
}
pub fn init_logger(
log_format: Option<&str>,
host_log_format: Option<&str>,
json: bool,
should_log: impl Fn(&Metadata) -> bool + Sync + Send + 'static,
) -> DefmtLoggerInfo {
let (logger, info): (Box<dyn Log>, DefmtLoggerInfo) = match json {
false => {
let logger = StdoutLogger::new(log_format, host_log_format, should_log);
let info = logger.info();
(logger, info)
}
true => {
JsonLogger::print_schema_version();
let logger = JsonLogger::new(log_format, host_log_format, should_log);
let info = logger.info();
(logger, info)
}
};
log::set_boxed_logger(logger).unwrap();
log::set_max_level(LevelFilter::Trace);
info
}
#[derive(Clone, Copy)]
pub struct DefmtLoggerInfo {
has_timestamp: bool,
}
impl DefmtLoggerInfo {
pub fn has_timestamp(&self) -> bool {
self.has_timestamp
}
}