prima-tracing 0.22.0

Prima.it opentelemetry standard library
Documentation
use opentelemetry::trace::Status;

use tracing::field::{Field, Visit};
use tracing::{Event, Level, Span, Subscriber};
use tracing_opentelemetry::OpenTelemetrySpanExt as _;
use tracing_subscriber::{registry::LookupSpan, Layer};

pub struct ErrorLayer;

impl<S> Layer<S> for ErrorLayer
where
    S: Subscriber + for<'span> LookupSpan<'span>,
{
    fn on_event(&self, event: &Event<'_>, ctx: tracing_subscriber::layer::Context<'_, S>) {
        // This is not an error
        if !event.metadata().is_event() || event.metadata().level() != &Level::ERROR {
            return;
        }

        // No current span, nothing to do
        if ctx.lookup_current().is_none() {
            return;
        }

        let mut visitor = ErrorVisitor::default();
        event.record(&mut visitor);

        if !visitor.is_error {
            return;
        }

        let span: Span = Span::current();

        // Tag Datadog: error.* as span attributes
        // See here for more info: https://docs.datadoghq.com/tracing/error_tracking/#use-span-attributes-to-track-error-spans
        span.set_attribute("error.type", visitor.kind.clone());
        span.set_attribute("error.message", visitor.message.clone());
        span.set_attribute("error.stack", visitor.stack.clone());

        // Optional but useful
        span.set_attribute("error", true);

        span.set_status(Status::error(visitor.message));
    }
}

#[derive(Default)]
struct ErrorVisitor {
    message: String,
    kind: String,
    stack: String,
    is_error: bool,
}

impl Visit for ErrorVisitor {
    fn record_error(&mut self, _field: &Field, value: &(dyn std::error::Error + 'static)) {
        let mut source: String = value.to_string();
        let mut next_err = value.source();

        while let Some(err) = next_err {
            source.push_str(&format!("\n{err}"));
            next_err = err.source();
        }

        let error_msg = value.to_string();

        self.message = error_msg;
        self.kind = "Error".to_string();
        self.stack = source;
        self.is_error = true;
    }

    fn record_debug(&mut self, _field: &Field, _value: &dyn std::fmt::Debug) {}
}