Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

Populate attributes from span tree? #88

Open
x1a0 opened this issue Jan 19, 2024 · 3 comments
Open

Populate attributes from span tree? #88

x1a0 opened this issue Jan 19, 2024 · 3 comments

Comments

@x1a0
Copy link

x1a0 commented Jan 19, 2024

Feature Request

Motivation

In tracing, I think it's common that the metadata about one trace are spread in different spans. Different layers in an application create spans, and tag the span with metadata they are responsible.

When a metric event is emitted, I'd like to be able to use those metadata that could be found in the span tree. Either flatten and keep all attributes in the tree and attach them to the metric event, or, even better, being able to somehow filter the attributes.

@ymgyt
Copy link
Contributor

ymgyt commented Jan 20, 2024

how about this ?

  1. Add an option to MetricsLayer to enable collection of attributes from span field.
  2. Add a closure to allow filtering of fields to be added to the metrics attribute from
MetrycsLayer::new(provider)
  .with_collect_span(true)
  .with_collect_span_filter(| field: &tracing::field::Field| -> bool { field.name() == "my_interest" })

If span collect is enabled, MetricsLayer iterates span during metrics event processing and adds the field of span to metrics attribute.

@x1a0
Copy link
Author

x1a0 commented Jan 22, 2024

@ymgyt Thanks for the quick reply. The proposal looks good 👍

I am trying to implement it locally but facing a problem:

MetricsLayer iterates span during metrics event processing and adds the field of span to metrics attribute

I cannot figure out how to directly read the fields of spans when processing an event. I have some code like:

        if let Some(scope) = ctx.event_scope(event) {
            for span in scope.from_root() {
                // how to visit span's fields here...?
            }
        }

So far by reading other libraries, my learning is that I need to store the fields in span extensions, then I can read from the extensions when processing an event - is this the only way?

@ymgyt
Copy link
Contributor

ymgyt commented Jan 22, 2024

@x1a0

So far by reading other libraries, my learning is that I need to store the fields in span extensions, then I can read from the extensions when processing an event - is this the only way?

Yes, as I understand it, if you want to refer to the span's field later, you need to set it in the extension of each span.

        if let Some(scope) = ctx.event_scope(event) {
            for span in scope.from_root() {
                let mut extension = span.extensions_mut();
                let Some(metrics_layer_ext) = extension.get_mut::<MetricsLayerExtension>() else {
                  continue;
                }
                // get metrics attribute from metrics_layer_ext 
                // and insert here https://github.com/tokio-rs/tracing-opentelemetry/blob/b2013d085cee229c83eb1558c3263e536186fb89/src/metrics.rs#L404
            }
        }

There might be other ways, but as far as I know, that is the only way.

# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants