-
Notifications
You must be signed in to change notification settings - Fork 896
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Proposal: Tracer Components #10
Comments
Some more thoughts, should every tracer in the registry support full configuration? For instance different sampling, reporting mechanism. Perf-wise I would prefer a cached registry and also a single instrumentation might use multiple instrumentation points - e.g. servlet filer and jaxrs interceptors. This could be an iteration after #107 - if we allow a globally scoped tracer. |
Something I'd be quite interested to know, is how common an application needs such specific scenario. I remembering seeing this pattern in HTrace, but haven't seen it much in other places. Depending on that, I'd be up to putting this in out |
Love the use case. However, I'm not sure tracer-per-component is necessary to address that. For example, if instrumentation follows the OpenTracing conventions and sets the "component" tag on span builder, then the tracer can compare it with configuration and decide to return a noop span with span context equal to the parent. |
We currently name components in OpenTracing via a tag:
I agree there should be a This concept is more related to scope management and span creation; I don't feel it is related to the global tracer, or tracer instantiation. Tracers are a mechanism for recording data and exporting it - ideally, there should be only one. We should not conflate instantiating the export mechanism with name-spacing and scoping the current operations. Likewise, I would prefer not to conflate this kind of namespacing to a particular language feature, such as classes or functions – it should be a cross-language concept. This still leaves the question open as to where to put this information. I suggest that this information is part of the |
Am I understanding correctly that we want to have a notion of @tedsuo I don't think we should propagate this between spans, maybe require instrumentation to set this when they create new Spans. |
I can see pros and cons in inheriting component from parent span, but I think the con is bigger: if another layer doesn't set component then inheriting it is unexpected behavior. So I'd rather have it set on every span, which isn't too onerous for middleware for a given library/framework. |
Also a contrib |
@yurishkuro I'm not actually sure what the current inheritance behavior is with the @bogdandrutu that might be a good way to model the behavior @discostu105 was asking for. We could move it into core if it looks like a good pattern. Ideally, I would like "best practices" to not require contrib projects. |
I agree with @yurishkuro on not inheriting component from the parent |
This is a bit broader, but relevant. Please bear with me: In general, each trace event can be modeled as if it is occurring in a nested namespace. So far, I have seen a lot of interest in the following namespaces: Trace::Service::Component::Operation Currently state of affairs (correct me if I am wrong):
In the long run, I suspect we want something like Name, ID, and Attributes for each namespace. In the short run, I am hearing a lot of interest in attributes for components. Only having them on the span and the service is proving cumbersome for real world usage. So perhaps we should think deeper about this subject. |
Would some sort of |
I am not sure Name/ID/Attributes applies to all 4 entities:
service and component sound like two different levels of |
Thanks, I agree @yurishkuro, it's not clear that this needs to be fully normalized. But we have some holes, especially around attributes and the component scope. |
Instead of
|
I think developers want an API that lets them write concise instrumentation code, so that the weight of instrumentation doesn't overwhelm the code itself. The question is how individual modules of code will (concisely) declare their component in the source, at that point where spans are built. This suggests making it possible to build spans and supply very few "component" arguments, potentially just one argument at the call site. In my own code, I've accomplished this by bundling a set of attributes, including the component concept, into a single object that's passed to the span builder. Importantly, this includes more than one attribute potentially. The pattern generally involves some heavy-weight resource that is loaded into memory. It's part of a component, sure, but it also includes a number of other attributes that specifically describe an instance of the component being invoked at the point where a span is created. The code could read like:
The component builder would be a subset of the span builder API, merely allowing attribute names and setting the standard component attribute, for use in building spans elsewhere in the same module of code. |
It is an interesting pattern, I think the instrumentation could just store I liked |
Thanks @jmacd. I discussed this with @pavolloffay, @bogdandrutu, and @carlosalberto. An updated proposal is as follows: class MyLibrary {
Tracer tracer;
MyLibrary() {
tracer = OpenTelemetry.getTracer().toBuilder("this_is_my_class").
setResource(resource).
build();
// ...
}
void someMethod(String arg) {
Span span = this.tracer.spanBuilder("someMethod").setAttribute("arg", arg).startSpan()
// ...
}
}; Main Notes:
Additional Notes:
|
(BTW I don't know why Github keeps putting @pavolloffay's comment at the bottom of this thread, but it is out of order.) |
@tedsuo see the attachment :))), @pavolloffay came from the future :))) |
I like it, @tedsuo. I always focus on concision, so I wonder if a ResourceBuilder might help with readability. For example, to construct a resource in the same line of code, you might like to write:
|
@jmacd that is not really a ResourceBuilder but I think offering a simpler API to create Resource (or even a real builder) should be on our list. |
From our (Dynatrace) perspective, the component should not be optional. OTel users should be enforced to provide a component name.
of course: the whole builder is optional from component perspective. |
I think there was a bit of a mixup of my intents of this "component" (sorry for late reply). Maybe the name "component" is misleading. I actually did not mean to identify "the component being traced", but rather "the tracing-implementation". That's why the existing "component" tag from OpenTracing is not really what I meant here. So, IMHO this should not be conflated with "resources", as these are in my understanding, rather identifying something like a host/process/service/library. Knowing the instrumentation-component for every span would allow our agent to determine which instrumentation-components are active in an application at runtime. Based on that we could, at runtime, decide which instrumentation-components we would actually use. This could be configuration based (user opt-in per instrumentation-component) or we could disable specific instrumentations (e.g. faulty/bad instrumentation) based on such component information. Another use-case is, we could identify certain (well known) instrumentation-components that would overlap with our own already built-in auto-instrumentation and auto-blacklist them (or let the user decide which instrumentation to use) to avoid "double-tracing". Such concerns have also been brought up recently in the Auto-Instrumentation SIG (see https://docs.google.com/document/d/1sovSQIGdxXtsauxUNp4qUMEIJZzObdukzPT52eyPCHM/edit#heading=h.dcogdcm13u12)
|
Trying to get the Gist of the discussion again. We need to know two additional things about each span that gets generated:
In order to get this information into a span we have three options:
We could overcome this problem by creating a Does this capture the problem and proposed solutions properly? |
@AloisReitbauer this does capture the issue well. Modeling a The only complication is how the Tracer object relates context propagation. Both traces and metrics will want to be labelled with this information. So there's still some work to be done, in general, to cleanly separate the context propagation layer from the observability layer. |
+1 that this would be extremely useful to us over at Honeycomb - cc @maplebed We sometimes like to write to more than one distinct dataset or even telemetry processing environment from some of our internal tools. |
@z1c0 wrote an RFC that covers this issue: open-telemetry/oteps#16 |
I think this can be closed once #264 is merged. |
Closing this now. Thank you all, for taking this over the finish line. |
- Move CoC and contributing to templates directory - Create security template - Modify repository to address feedback Addresses open-telemetry#9 open-telemetry#10
Proposal: Instead of a global tracer singleton, every library needs to create it's own tracer, which also has a proper name that identifies the library/component that uses the tracer.
Motivation: "Configurability of library/component instrumentation"
Technically, this could be achieved similar to how logging frameworks deal with Loggers and LoggerFactories (log4j).
Every span would contain the information about the Tracer that it was created with. Based on that a span can be discarded or not.
Not sure how a naming convention could look like. Not sure if "component" is the right word, as it has already been used in the past as attribute name. At Dynatrace we call that a "Sensor". Not sure about granularity of "Tracers". Should every class have one? Should
Tracing
act as a registry (cache already created instances) or as a factory (just create new ones at every call)?This is kind of related to the GlobalTracer discussion (#107).
The text was updated successfully, but these errors were encountered: