about summary refs log tree commit diff
path: root/tvix/tracing/src
diff options
context:
space:
mode:
Diffstat (limited to 'tvix/tracing/src')
-rw-r--r--tvix/tracing/src/lib.rs110
1 files changed, 61 insertions, 49 deletions
diff --git a/tvix/tracing/src/lib.rs b/tvix/tracing/src/lib.rs
index 1305778a3d9a..3362818bb180 100644
--- a/tvix/tracing/src/lib.rs
+++ b/tvix/tracing/src/lib.rs
@@ -5,7 +5,11 @@ use tracing::level_filters::LevelFilter;
 use tracing_indicatif::{
     filter::IndicatifFilter, util::FilteredFormatFields, writer, IndicatifLayer, IndicatifWriter,
 };
-use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, EnvFilter, Layer};
+use tracing_subscriber::{
+    layer::{Identity, SubscriberExt},
+    util::SubscriberInitExt,
+    EnvFilter, Layer, Registry,
+};
 
 #[cfg(feature = "otlp")]
 use opentelemetry::{
@@ -175,74 +179,82 @@ impl TracingBuilder {
     /// It will also configure otlp if the feature is enabled and a service_name was provided. It
     /// will then correctly setup a channel which is later used for flushing the provider.
     pub fn build(self) -> Result<TracingHandle, Error> {
+        self.build_with_additional(Identity::new())
+    }
+
+    /// Similar to `build()` but allows passing in an additional tracing [`Layer`].
+    ///
+    /// This method is generic over the `Layer` to avoid the runtime cost of dynamic dispatch.
+    /// While it only allows passing a single `Layer`, it can be composed of multiple ones:
+    ///
+    /// ```ignore
+    /// build_with_additional(
+    ///   fmt::layer()
+    ///     .and_then(some_other_layer)
+    ///     .and_then(yet_another_layer)
+    ///     .with_filter(my_filter)
+    /// )
+    /// ```
+    /// [`Layer`]: tracing_subscriber::layer::Layer
+    pub fn build_with_additional<L>(self, additional_layer: L) -> Result<TracingHandle, Error>
+    where
+        L: Layer<Registry> + Send + Sync + 'static,
+    {
         // Set up the tracing subscriber.
         let indicatif_layer = IndicatifLayer::new().with_progress_style(PB_SPINNER_STYLE.clone());
         let stdout_writer = indicatif_layer.get_stdout_writer();
         let stderr_writer = indicatif_layer.get_stderr_writer();
-        let subscriber = tracing_subscriber::registry()
-            .with(
-                EnvFilter::builder()
-                    .with_default_directive(LevelFilter::INFO.into())
-                    .from_env()
-                    .expect("invalid RUST_LOG"),
-            )
-            .with(
-                tracing_subscriber::fmt::Layer::new()
-                    .fmt_fields(FilteredFormatFields::new(
-                        tracing_subscriber::fmt::format::DefaultFields::new(),
-                        |field| field.name() != "indicatif.pb_show",
-                    ))
-                    .with_writer(indicatif_layer.get_stderr_writer())
-                    .compact(),
-            )
-            .with((self.progess_bar).then(|| {
+
+        let layered = tracing_subscriber::fmt::Layer::new()
+            .fmt_fields(FilteredFormatFields::new(
+                tracing_subscriber::fmt::format::DefaultFields::new(),
+                |field| field.name() != "indicatif.pb_show",
+            ))
+            .with_writer(indicatif_layer.get_stderr_writer())
+            .compact()
+            .and_then((self.progess_bar).then(|| {
                 indicatif_layer.with_filter(
                     // only show progress for spans with indicatif.pb_show field being set
                     IndicatifFilter::new(false),
                 )
             }));
+        #[cfg(feature = "tracy")]
+        let layered = layered.and_then(TracyLayer::default());
+
+        let mut tx: Option<mpsc::Sender<Option<oneshot::Sender<()>>>> = None;
 
         // Setup otlp if a service_name is configured
         #[cfg(feature = "otlp")]
-        {
+        let layered = layered.and_then({
             if let Some(service_name) = self.service_name {
                 // register a text map propagator for trace propagation
                 opentelemetry::global::set_text_map_propagator(TraceContextPropagator::new());
 
-                let (tracer, tx) = gen_otlp_tracer(service_name.to_string());
+                let (tracer, sender) = gen_otlp_tracer(service_name.to_string());
+                tx = Some(sender);
                 // Create a tracing layer with the configured tracer
-                let layer = tracing_opentelemetry::layer().with_tracer(tracer);
-
-                #[cfg(feature = "tracy")]
-                {
-                    subscriber
-                        .with(TracyLayer::default())
-                        .with(Some(layer))
-                        .try_init()?;
-                }
-
-                #[cfg(not(feature = "tracy"))]
-                {
-                    subscriber.with(Some(layer)).try_init()?;
-                }
-                return Ok(TracingHandle {
-                    tx: Some(tx),
-                    stdout_writer,
-                    stderr_writer,
-                });
+                Some(tracing_opentelemetry::layer().with_tracer(tracer))
+            } else {
+                None
             }
-        }
-        #[cfg(feature = "tracy")]
-        {
-            subscriber.with(TracyLayer::default()).try_init()?;
-        }
-        #[cfg(not(feature = "tracy"))]
-        {
-            subscriber.try_init()?;
-        }
+        });
+
+        let layered = layered.with_filter(
+            EnvFilter::builder()
+                .with_default_directive(LevelFilter::INFO.into())
+                .from_env()
+                .expect("invalid RUST_LOG"),
+        );
+
+        tracing_subscriber::registry()
+            // TODO: if additional_layer has global filters, there is a risk that it will disable the "default" ones,
+            // while it could be solved by registering `additional_layer` last, it requires boxing `additional_layer`.
+            .with(additional_layer)
+            .with(layered)
+            .try_init()?;
 
         Ok(TracingHandle {
-            tx: None,
+            tx,
             stdout_writer,
             stderr_writer,
         })