-
Notifications
You must be signed in to change notification settings - Fork 437
/
context.go
62 lines (53 loc) · 2.26 KB
/
context.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
// Unless explicitly stated otherwise all files in this repository are licensed
// under the Apache License Version 2.0.
// This product includes software developed at Datadog (https://www.datadoghq.com/).
// Copyright 2016 Datadog, Inc.
package tracer
import (
"context"
"gopkg.in/DataDog/dd-trace-go.v1/ddtrace"
"gopkg.in/DataDog/dd-trace-go.v1/ddtrace/internal"
)
type contextKey struct{}
var activeSpanKey = contextKey{}
// ContextWithSpan returns a copy of the given context which includes the span s.
func ContextWithSpan(ctx context.Context, s Span) context.Context {
return context.WithValue(ctx, activeSpanKey, s)
}
// SpanFromContext returns the span contained in the given context. A second return
// value indicates if a span was found in the context. If no span is found, a no-op
// span is returned.
func SpanFromContext(ctx context.Context) (Span, bool) {
if ctx == nil {
return &internal.NoopSpan{}, false
}
v := ctx.Value(activeSpanKey)
if s, ok := v.(ddtrace.Span); ok {
return s, true
}
return &internal.NoopSpan{}, false
}
// StartSpanFromContext returns a new span with the given operation name and options. If a span
// is found in the context, it will be used as the parent of the resulting span. If the ChildOf
// option is passed, it will only be used as the parent if there is no span found in `ctx`.
func StartSpanFromContext(ctx context.Context, operationName string, opts ...StartSpanOption) (Span, context.Context) {
// copy opts in case the caller reuses the slice in parallel
// we will add at least 1, at most 2 items
optsLocal := make([]StartSpanOption, len(opts), len(opts)+2)
copy(optsLocal, opts)
if ctx == nil {
// default to context.Background() to avoid panics on Go >= 1.15
ctx = context.Background()
} else if s, ok := SpanFromContext(ctx); ok {
optsLocal = append(optsLocal, ChildOf(s.Context()))
}
optsLocal = append(optsLocal, withContext(ctx))
s := StartSpan(operationName, optsLocal...)
if span, ok := s.(*span); ok && span.pprofCtxActive != nil {
// If pprof labels were applied for this span, use the derived ctx that
// includes them. Otherwise a child of this span wouldn't be able to
// correctly restore the labels of its parent when it finishes.
ctx = span.pprofCtxActive
}
return s, ContextWithSpan(ctx, s)
}