use query_core::telemetry::helpers as telemetry_helpers;
use query_engine_metrics::MetricRegistry;
use tracing_error::ErrorLayer;
use tracing_subscriber::{layer::Layered, prelude::*, Layer, Registry};
use crate::LogEmit;
type Sub = Layered<
ErrorLayer<
Layered<
Box<
dyn tracing_subscriber::Layer<
Layered<Box<dyn tracing_subscriber::Layer<Registry> + Send + Sync>, Registry>,
> + Send
+ Sync,
>,
Layered<Box<dyn tracing_subscriber::Layer<Registry> + Send + Sync>, Registry>,
>,
>,
Layered<
Box<
dyn tracing_subscriber::Layer<Layered<Box<dyn tracing_subscriber::Layer<Registry> + Send + Sync>, Registry>>
+ Send
+ Sync,
>,
Layered<Box<dyn tracing_subscriber::Layer<Registry> + Send + Sync>, Registry>,
>,
>;
pub fn test_tracing_subscriber(log_config: String, metrics: MetricRegistry, log_tx: LogEmit) -> Sub {
let filter = telemetry_helpers::env_filter(true, telemetry_helpers::QueryEngineLogLevel::Override(log_config));
let fmt_layer = tracing_subscriber::fmt::layer()
.with_writer(PrintWriter::new(log_tx))
.with_filter(filter);
tracing_subscriber::registry()
.with(fmt_layer.boxed())
.with(metrics.boxed())
.with(ErrorLayer::default())
}
pub struct PrintWriter {
tx: LogEmit,
}
impl PrintWriter {
fn new(tx: LogEmit) -> Self {
Self { tx }
}
}
impl tracing_subscriber::fmt::MakeWriter<'_> for PrintWriter {
type Writer = PrintWriter;
fn make_writer(&self) -> Self::Writer {
PrintWriter::new(self.tx.clone())
}
}
impl std::io::Write for PrintWriter {
fn write(&mut self, buf: &[u8]) -> Result<usize, std::io::Error> {
let log = std::str::from_utf8(buf).unwrap_or("<invalid UTF-8>");
if log.contains("quaint") || log.contains("mongodb_query_connector") {
let plain_bytes = strip_ansi_escapes::strip(buf)?;
let plain_log = std::str::from_utf8(&plain_bytes).unwrap_or("<invalid UTF-8>");
let _ = self.tx.send(plain_log.to_string());
}
eprint!("{log}");
Ok(buf.len())
}
fn flush(&mut self) -> Result<(), std::io::Error> {
Ok(())
}
}