1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
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;

// Pretty ugly. I'm not sure how to make this better
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())
}

/// This is a temporary implementation detail for `tracing` logs in tests.
/// Instead of going through `std::io::stderr`, it goes through the specific
/// local stderr handle used by `eprintln` and `dbg`, allowing logs to appear in
/// specific test outputs for readability.
///
/// It is used from test_macros.
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(())
    }
}