-
Notifications
You must be signed in to change notification settings - Fork 897
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
Context interaction in different packages like trace/baggage is unclear #1019
Comments
Only the first two are about installing a Span in the Context. I assumed the operations were removed from "tracer-operations" when they were added to the new "tracing-context-utilities", maybe it is just a mixup that they still exist in the "tracer-operations"? If it is removed from "tracer-operations" then there is only 1 documented way to do it. |
They are to install a Span into an "active/current Context", but we still lack a
That may be good, but was thinking that we can postpone this and think more, once we have APIs to interact with a |
@bogdandrutu I advise you to add examples of how you would use the proposed API. Without examples it is difficult to fully understand the proposal. I can assume/imagine how it would be used, but my assumptions may be wrong. Examples will help facilitate the discussion. Show a few use cases. |
I don't think there should be a -spec w3c_propagators() -> {otel_propagation:http_extractor(), otel_propagation:http_injector()}.
w3c_propagators() ->
ToText = fun otel_propagation_http_w3c:inject/2,
FromText = fun otel_propagation_http_w3c:extract/2,
Injector = otel_ctx:http_injector(?TRACER_CTX, ToText),
Extractor = otel_ctx:http_extractor(?EXTERNAL_SPAN_CTX, FromText),
{Extractor, Injector}. So for injecting a different Context key is used ( |
@tsloughter (I told you to not post more erlang code because I find bugs):
|
@tigrannajaryan the current Java API example (we still use public interface Tracer {
Span getCurrentSpan();
// Scope is just an auto-closable instance.
Scope withSpan(Span span);
}
public final class TracingContextUtils {
private static final Context.Key<Span> CONTEXT_SPAN_KEY =
Context.key("opentelemetry-trace-span-key");
// THIS IS NOT IN THE SPECS NOW
public static Context withSpan(Span span, Context context) {
return context.withValue(CONTEXT_SPAN_KEY, span);
}
public static Span getCurrentSpan() {
return getSpan(Context.current());
}
// THIS IS NOT IN THE SPECS NOW
public static Span getSpan(Context context) {
Span span = CONTEXT_SPAN_KEY.get(context);
return span == null ? DefaultSpan.getInvalid() : span;
}
public static Scope currentContextWith(Span span) {
return ContextUtils.withScopedContext(withSpan(span, Context.current()));
}
private TracingContextUtils() {}
}
public final class CurrentContext {
public static Span getCurrent() {
...
}
public static Token attach() {
...
}
public static detach(t Token) {
...
}
public static Scope withContext(Context context);
} So based on the current API (defined by the specs), I can get the active Span in 3 ways:
EDITED Indeed as @tsloughter pointed the |
@bogdandrutu the keys aren't exposed to the user, they are macros internal to the module only. |
And it looks like 2 keys can still be used, it is just named |
@bogdandrutu if I read your proposal correctly you suggest to remove the first 2 ways and only allow the last one. That looks like a reasonable, composable API to me. I would still want to see API usage examples. In my mind there are a few important use cases:
There are probably more. It would help a lot to see what is the shape of the code for these uses cases with your proposed API. I generally agree with you that it is bad to have 3 slightly different ways to achieve the same thing. |
|
The keys don't have to be used directly by the user to implement a propagator (at least in this implementation). The keys are used by But damn, didn't read that PR right... Don't think I like a new form of Span for this, but I guess easy enough to switch to. It feels like a lot of stuff has been merged quickly in the last couple days in order to meet the freeze even though it changes stuff that had gone through a lot of discussion already, without similar discussion time. |
@tsloughter completely agree with your assessment regarding merging things. I was keep saying this in a lot of meetings that I am worried about. Regarding |
Right, this is just for the "Text" propagator -- hasn't been renamed since it was renamed in the spec. There is Text and Binary, right? |
@tigrannajaryan I will try to shape these examples here (mostly copying from Java codebase):
// Wrapper for remote calls.
public void interceptCall(args) {
TextMapPropagator textFormat = OpenTelemetry.getPropagators().getTextMapPropagator();;
// Extract the context entries from the metadata of the gRPC request
Context extractedContext = textFormat.extract(Context.current(), headers, getter);
Span span =
tracer
.spanBuilder("helloworld.Greeter/SayHello")
.setParent(extractedContext)
.setSpanKind(Span.Kind.SERVER)
.startSpan();
// Here a helper like TracingContextUtils.currentContextWith will remove this extra line
appContext = TracingContextUtils.withSpan(extractedContext, span)
try (Scope scope = CurrentContext.withContext(appContext)) {
doApplicationWork(args)
} finally {
span.end()
}
}
public void doApplicationWork(args) {
// Here a helper like TracingContextUtils.getCurrentSpan will remove this extra call to CurrentContext.getCurrent()
Span currentSpan = TracingContextUtils.getSpan(CurrentContext.getCurrent())
// Do some work
currentSpan.setAttribute("foo", "bar")
// Do more work
// Create a Span with the initial server span as a parent.
Span span = tracer.spanBuilder("ScheduleAsyncWork").startSpan();
try {
// Here a helper like TracingContextUtils.currentContextWith will remove this extra line
context = TracingContextUtils.withSpan(CurrentContext.getCurrent(), span)
// Here it is important to wrap the Runnable with the entire context not just trace Span.
executor.execute(CurrentContext.wrap(context, doAsyncApplicationWork(args))))
} finally {
span.end()
}
}
public Runnable doAsyncApplicationWork(args) {
return new Runnable {
@Override
public void run() {
// Here a helper like TracingContextUtils.getCurrentSpan will remove this extra call to CurrentContext.getCurrent()
Span currentSpan = TracingContextUtils.getSpan(CurrentContext.getCurrent())
currentSpan.AddEvent("Happily executing the async work")
// Create a Span with the "ScheduleAsyncWork" span as a parent.
Span span = tracer.spanBuilder("ScheduleAsyncWork").startSpan();
// Here a helper like TracingContextUtils.currentContextWith will remove this extra line
Context context = TracingContextUtils.withSpan(CurrentContext.getCurrent(), span)
try (Scope scope = CurrentContex.withContext(context) {
// Here the new Span is active.
} finally {
span.end()
}
}
}
} EDITED |
@bogdandrutu do you think we need to postpone the trace spec freeze until we resolve this? We may need a bit time to do that. |
If I read this correctly, there will only be one method left on the Tracer interface (in Java, at least), which is to create a new I definitely agree that things are muddled at the moment, and it's unclear what the "right" way to do manage the Context is. But, I'm not sure if completely gutting the Tracer is the right way to approach it. I feel like, as an API user, that having the Tracer be the place to do "tracing", which by its very nature involves having to manage the propagation of the context, feels like a more natural approach. My instincts as an API designer tell me that we should make the Tracer the one-stop-shop for doing tracing, both creating spans, and propagating them, rather than moving all of that to the more wonkily named "TracingContextUtils". I do like the idea of a CurrentContext class that is the place for dealing with the context itself, outside of tracing concerns, although I might just want to call it a ContextManager at that point. Those are my rough thoughts at this point. And, yes, I don't think this is something we can just rush through on a Friday night. :) |
@tigrannajaryan these are similar to (more complementary to, I guess) the examples I'm hoping to get for languages, https://docs.google.com/document/d/1kEW71SYInRB44elJl116vrzYL3_HXr133JomMokZ-PI/edit# Maybe there should be a checkbox per example in the GA spec table? They are going to be useful not just for discussing and finalizing the API but for users as documentation as well. |
I have been brainstorming integrating trace context methods into static Span.current()
Scope Span.makeCurrent()
static Span fromContext(Context ctx)
Context addToContext(Context ctx) I think this could be a simpler Java API where static util methods on an unrelated class are not so idiomatic - when working with a span, use the span. Or the context. But it's not common to have a third # party do things I think. Does the spec preclude merging pieces together (Span and TraceContextUtils) in this sort of way? |
@anuraaga I don't think you can implement the Which means you have the same functionality in the Span class instead of the helper. I would say at that point that becomes only a personal preference. Also this issue is not about necessary where the functionality lives, it is about duplicate exposed functionality and missing functionality first, then about where to expose it. That being said I think your proposal should be consider as one of the possibility to expose the functionality. |
Ah for both of these non-static methods, I'm thinking of using default methods. If the key is package private, I think it'll not have that problem.
But this is just what I wanted to confirm :) Sounds good then. |
I made an example/test propagator to hopefully better show how we are doing it with anonymous functions in anonymous functions: https://github.com/open-telemetry/opentelemetry-erlang/blob/0188ee39790f91df8a3a106beea9cf2fddd5af5e/apps/opentelemetry_api/test/custom_propagator.erl#L41 For the injector the context key is passed along with a function that will return the content to include in the "text map". For the extractor the carrier and the current value of the key in the context are passed to the function and the returned updated value is inserted into the context by the helper. So the key isn't exposed but it is passed to and stored in the list of propagators. |
+1 on this comment from @jkwatson - I think I mentioned it in the past, but in OpenTracing we still had @bogdandrutu I see the reason to have less methods, that indeed may create confusion, but I'd be up for:
|
+1 to this +1. I think this is also a case of implementation bleeding into spec. "Utils" is used to simply describe functions that are not dependent to the stateful "Tracer", right? |
That's an interesting interpretation. So, in Java, for example, you think it would be fine to have these methods just be static methods on the Tracer class. That sounds much better to me than what we have at the moment. I think that could use some clarification, if generally agreed upon. |
My 2 cents is that span / context interactions should be handled through the tracer. If the operations don't require a specific tracer instance, then they can be static methods on the tracer. For Ruby, we have two methods, that are working well for us.
I don't know that we need to introduce a new namespace such as ContextUtils for these methods. Having them on the tracer would be in line with the remove span proposal: open-telemetry/oteps#68. |
Regarding the context propagation and not exposing keys: Currently propagators are spec'ed and in most languages written as the individual propagator implementations (like b3, w3c tracecontext and w3c baggage) must handle Context. I'm arguing that propagators are per-concern (tracing, baggage, etc) and this is separate from the propagator implementation. Meaning a Propagator still takes a Context but w3c and b3 implementations are serialization functions passed to the Propagator, not Propagators themselves.
This boils down to what today are called Propagators become "serializers" (or whatever) and a Propagator is instead a specific instance of a TextMapPropagator that is created by Tracer or Baggage (each which know the context key to get the particular value they are injecting from or extracting to). |
@bogdandrutu I noticed you wrote "THIS IS NOT IN THE SPECS NOW" twice in though I would point out that these APIs to set and get the current span exist in the OTel-Go library and they feel natural. I support this aspect of your proposal. I believe this idea has been previously referred to as the idea that Spans be created in the "detached" state, forcing you to use In the OTel-Go SDK there are three context keys (baggage, span context, remote span context). Are you hoping to reduce the number of context keys overall in the move to keep them in one place (baggage and remote span context are often updated together)? From reading the above comments, it looks like there is support for keeping these Context methods in the trace package, not a "Utils" package. Maybe baggage should move back into trace, so that all these APIs are in the same place? |
Baggage should remain separate, it could be utilized even without traces but just to propagate some baggage or use it with metrics. |
@jmacd also, apparently a separate context key isn't supposed to be used for remote extracted spans anymore? This is what Bogdan was saying in this issue. Is Go still using the separate key? Erlang is. Go is also not OOP so wondering if "PropagatedSpan" doesn't work well in it either. |
from the spec sig mtg today, will assess prototypes by next spec sig in order to move on a decision |
PS: This is written as a last moment issue before GA, and may not be 100% cleared. But it raises a concern that we cannot GA without understanding and fixing it.
What are you trying to achieve?
Simplify API, and consistency across packages. Issue #455 was closed without fixing this problem.
What did you expect to see?
Fewer APIs that do the same thing.
Additional context
Right now, as defined by the specs there are multiple ways to install a
Span
in the activeContext
, and some require aTracer
instance, or some require the globalContext
instance:Besides the fact that currently there are multiple ways to get the current Context (or current
Span
), the specs does not provide a way to extract aSpan
form a givenContext
object.Proposal
The current proposal is to separate the
Context
interaction into two parts:Context
instance.This functionality is important for
Propagators
and forTracer
implementations. User can also use this functionality when manually propagating aContext
object.This is a per package concern,
trace
andbaggage
must expose a way to achieve this. It is recommended to not directly expose theContext.Key
, see here.Context
propagation mechanism (such as a thread local implementation).This functionality is important only for languages that support this, and it is important to be consistent across implementations, as well as be able to do propagation without requiring a
Tracer
orBaggageManager
.Tracer implementation may need access to this mechanism to implement the "implicit parent from context" functionality.
The proposed solution is:
Context
instance.":WithSpan
andFromContext
(names to be decided).Context
propagation mechanism.":Tracer
andTrace Context Utilities
that interact with "current Context".Things like
Get the currently active span
andSet the currently active span
should be removed.Get the currently active span
andSet the currently active span
: the context API to get the activeContext
and the helper to extract/add aSpan
to theContext
Alternative Consideration
Don't do anything
This is not going to work because of missing functionality and user confusion by having multiple ways to achieve the same thing.
Try to just add "Adding/Extracting a new entry to a given
Context
instance."This may help with the missing functionality that currently is not defined in the specs, but will increase the user's confusion when it comes to interacting with the
Context
and otel APIs.Keep provide helper functionality like
Get the currently active span
andSet the currently active span
The reason to remove this, is because currently they are defined in two places (with a may) which makes things inconsistent across different languages, and also proves that we do not have a good understanding of where this helpers should leave. Because of the imminent GA deadline, I suggest to remove this and add them later (adding new functionality is backwards compatible).
The text was updated successfully, but these errors were encountered: