mod json_logger;
mod pretty_logger;
use log::{Level, LevelFilter, Metadata, Record};
use serde::{Deserialize, Serialize};
use std::fmt;
use self::{json_logger::JsonLogger, pretty_logger::PrettyLogger};
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(
always_include_location: bool,
json: bool,
should_log: impl Fn(&Metadata) -> bool + Sync + Send + 'static,
) {
log::set_boxed_logger(match json {
false => PrettyLogger::new(always_include_location, should_log),
true => {
JsonLogger::print_schema_version();
JsonLogger::new(should_log)
}
})
.unwrap();
log::set_max_level(LevelFilter::Trace);
}