use std::borrow::ToOwned;
use std::sync::Arc;
use std::thread;
use backtrace::Backtrace;
use crossbeam_channel::Sender;
use log::{Level, LevelFilter, Log, Metadata, Record};
use parking_lot::ReentrantMutex;
use servo_base::id::{ScriptEventLoopId, TEST_PIPELINE_ID, TEST_WEBVIEW_ID};
use servo_constellation_traits::{
EmbedderToConstellationMessage, LogEntry, ScriptToConstellationMessage,
ScriptToConstellationSender,
};
pub struct FromScriptLogger {
pub script_to_constellation_sender: Arc<ReentrantMutex<ScriptToConstellationSender>>,
}
impl FromScriptLogger {
pub fn new(script_to_constellation_chan: ScriptToConstellationSender) -> FromScriptLogger {
FromScriptLogger {
script_to_constellation_sender: Arc::new(ReentrantMutex::new(
script_to_constellation_chan,
)),
}
}
pub fn filter(&self) -> LevelFilter {
LevelFilter::Warn
}
}
impl Log for FromScriptLogger {
fn enabled(&self, metadata: &Metadata) -> bool {
metadata.level() <= Level::Warn
}
fn log(&self, record: &Record) {
if let Some(entry) = log_entry(record) {
let thread_name = thread::current().name().map(ToOwned::to_owned);
let _ = self.script_to_constellation_sender.lock().send(
(
TEST_WEBVIEW_ID,
TEST_PIPELINE_ID,
ScriptToConstellationMessage::LogEntry(
ScriptEventLoopId::installed(),
thread_name,
entry,
),
),
);
}
}
fn flush(&self) {}
}
#[derive(Clone)]
pub struct FromEmbedderLogger {
pub constellation_chan: Arc<ReentrantMutex<Sender<EmbedderToConstellationMessage>>>,
}
impl FromEmbedderLogger {
pub fn new(constellation_chan: Sender<EmbedderToConstellationMessage>) -> FromEmbedderLogger {
FromEmbedderLogger {
constellation_chan: Arc::new(ReentrantMutex::new(constellation_chan)),
}
}
pub fn filter(&self) -> LevelFilter {
LevelFilter::Warn
}
}
impl Log for FromEmbedderLogger {
fn enabled(&self, metadata: &Metadata) -> bool {
metadata.level() <= Level::Warn
}
fn log(&self, record: &Record) {
if let Some(entry) = log_entry(record) {
let event_loop_id = ScriptEventLoopId::installed();
let thread_name = thread::current().name().map(ToOwned::to_owned);
let msg = EmbedderToConstellationMessage::LogEntry(event_loop_id, thread_name, entry);
let chan = self.constellation_chan.lock();
let _ = chan.send(msg);
}
}
fn flush(&self) {}
}
fn log_entry(record: &Record) -> Option<LogEntry> {
match record.level() {
Level::Error if thread::panicking() => Some(LogEntry::Panic(
format!("{}", record.args()),
format!("{:?}", Backtrace::new()),
)),
Level::Error => Some(LogEntry::Error(format!("{}", record.args()))),
Level::Warn => Some(LogEntry::Warn(format!("{}", record.args()))),
_ => None,
}
}