2020-03-23 22:41:10 -07:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2024-02-29 07:05:28 +01:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2019-08-02 13:52:55 -07:00
|
|
|
|
2020-11-04 18:10:58 +01:00
|
|
|
package trace // import "go.opentelemetry.io/otel/sdk/trace"
|
2019-08-02 13:52:55 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-09-02 14:30:12 -07:00
|
|
|
"time"
|
2019-08-02 13:52:55 -07:00
|
|
|
|
2020-06-09 11:47:54 -07:00
|
|
|
"go.opentelemetry.io/otel/sdk/instrumentation"
|
2021-11-13 08:35:04 -08:00
|
|
|
"go.opentelemetry.io/otel/trace"
|
2023-10-19 10:16:24 -07:00
|
|
|
"go.opentelemetry.io/otel/trace/embedded"
|
2019-08-02 13:52:55 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
type tracer struct {
|
2023-10-19 10:16:24 -07:00
|
|
|
embedded.Tracer
|
|
|
|
|
2022-07-06 11:55:46 -07:00
|
|
|
provider *TracerProvider
|
|
|
|
instrumentationScope instrumentation.Scope
|
2019-08-02 13:52:55 -07:00
|
|
|
}
|
|
|
|
|
2020-11-06 23:13:31 +01:00
|
|
|
var _ trace.Tracer = &tracer{}
|
2019-08-02 13:52:55 -07:00
|
|
|
|
2020-09-03 07:34:36 -07:00
|
|
|
// Start starts a Span and returns it along with a context containing it.
|
|
|
|
//
|
|
|
|
// The Span is created with the provided name and as a child of any existing
|
|
|
|
// span context found in the passed context. The created Span will be
|
2021-09-02 14:30:12 -07:00
|
|
|
// configured appropriately by any SpanOption passed.
|
2021-05-27 09:53:56 -05:00
|
|
|
func (tr *tracer) Start(ctx context.Context, name string, options ...trace.SpanStartOption) (context.Context, trace.Span) {
|
|
|
|
config := trace.NewSpanStartConfig(options...)
|
2019-08-02 13:52:55 -07:00
|
|
|
|
2022-08-26 07:53:33 -06:00
|
|
|
if ctx == nil {
|
|
|
|
// Prevent trace.ContextWithSpan from panicking.
|
|
|
|
ctx = context.Background()
|
|
|
|
}
|
|
|
|
|
2021-03-25 14:36:39 +00:00
|
|
|
// For local spans created by this SDK, track child span count.
|
2020-11-06 23:13:31 +01:00
|
|
|
if p := trace.SpanFromContext(ctx); p != nil {
|
2021-09-02 14:30:12 -07:00
|
|
|
if sdkSpan, ok := p.(*recordingSpan); ok {
|
2020-02-04 17:55:03 +01:00
|
|
|
sdkSpan.addChild()
|
2019-10-14 13:20:03 -07:00
|
|
|
}
|
2019-08-02 13:52:55 -07:00
|
|
|
}
|
|
|
|
|
2021-09-02 14:30:12 -07:00
|
|
|
s := tr.newSpan(ctx, name, &config)
|
|
|
|
if rw, ok := s.(ReadWriteSpan); ok && s.IsRecording() {
|
2023-04-02 01:57:35 +11:00
|
|
|
sps := tr.provider.getSpanProcessors()
|
2020-09-24 14:43:23 -04:00
|
|
|
for _, sp := range sps {
|
2021-09-02 14:30:12 -07:00
|
|
|
sp.sp.OnStart(ctx, rw)
|
2019-09-16 13:58:15 -07:00
|
|
|
}
|
|
|
|
}
|
2021-09-02 14:30:12 -07:00
|
|
|
if rtt, ok := s.(runtimeTracer); ok {
|
|
|
|
ctx = rtt.runtimeTrace(ctx)
|
|
|
|
}
|
2019-09-16 13:58:15 -07:00
|
|
|
|
2021-09-02 14:30:12 -07:00
|
|
|
return trace.ContextWithSpan(ctx, s), s
|
|
|
|
}
|
|
|
|
|
|
|
|
type runtimeTracer interface {
|
|
|
|
// runtimeTrace starts a "runtime/trace".Task for the span and
|
|
|
|
// returns a context containing the task.
|
|
|
|
runtimeTrace(ctx context.Context) context.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
// newSpan returns a new configured span.
|
|
|
|
func (tr *tracer) newSpan(ctx context.Context, name string, config *trace.SpanConfig) trace.Span {
|
|
|
|
// If told explicitly to make this a new root use a zero value SpanContext
|
|
|
|
// as a parent which contains an invalid trace ID and is not remote.
|
|
|
|
var psc trace.SpanContext
|
|
|
|
if config.NewRoot() {
|
|
|
|
ctx = trace.ContextWithSpanContext(ctx, psc)
|
|
|
|
} else {
|
|
|
|
psc = trace.SpanContextFromContext(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there is a valid parent trace ID, use it to ensure the continuity of
|
|
|
|
// the trace. Always generate a new span ID so other components can rely
|
|
|
|
// on a unique span ID, even if the Span is non-recording.
|
|
|
|
var tid trace.TraceID
|
|
|
|
var sid trace.SpanID
|
|
|
|
if !psc.TraceID().IsValid() {
|
|
|
|
tid, sid = tr.provider.idGenerator.NewIDs(ctx)
|
|
|
|
} else {
|
|
|
|
tid = psc.TraceID()
|
|
|
|
sid = tr.provider.idGenerator.NewSpanID(ctx, tid)
|
|
|
|
}
|
|
|
|
|
|
|
|
samplingResult := tr.provider.sampler.ShouldSample(SamplingParameters{
|
|
|
|
ParentContext: ctx,
|
|
|
|
TraceID: tid,
|
|
|
|
Name: name,
|
|
|
|
Kind: config.SpanKind(),
|
|
|
|
Attributes: config.Attributes(),
|
|
|
|
Links: config.Links(),
|
|
|
|
})
|
|
|
|
|
|
|
|
scc := trace.SpanContextConfig{
|
|
|
|
TraceID: tid,
|
|
|
|
SpanID: sid,
|
|
|
|
TraceState: samplingResult.Tracestate,
|
|
|
|
}
|
|
|
|
if isSampled(samplingResult) {
|
|
|
|
scc.TraceFlags = psc.TraceFlags() | trace.FlagsSampled
|
|
|
|
} else {
|
|
|
|
scc.TraceFlags = psc.TraceFlags() &^ trace.FlagsSampled
|
|
|
|
}
|
|
|
|
sc := trace.NewSpanContext(scc)
|
|
|
|
|
|
|
|
if !isRecording(samplingResult) {
|
|
|
|
return tr.newNonRecordingSpan(sc)
|
|
|
|
}
|
|
|
|
return tr.newRecordingSpan(psc, sc, name, samplingResult, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
// newRecordingSpan returns a new configured recordingSpan.
|
|
|
|
func (tr *tracer) newRecordingSpan(psc, sc trace.SpanContext, name string, sr SamplingResult, config *trace.SpanConfig) *recordingSpan {
|
|
|
|
startTime := config.Timestamp()
|
|
|
|
if startTime.IsZero() {
|
|
|
|
startTime = time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
s := &recordingSpan{
|
2022-02-07 12:58:05 -08:00
|
|
|
// Do not pre-allocate the attributes slice here! Doing so will
|
|
|
|
// allocate memory that is likely never going to be used, or if used,
|
|
|
|
// will be over-sized. The default Go compiler has been tested to
|
|
|
|
// dynamically allocate needed space very well. Benchmarking has shown
|
|
|
|
// it to be more performant than what we can predetermine here,
|
|
|
|
// especially for the common use case of few to no added
|
|
|
|
// attributes.
|
|
|
|
|
2022-02-01 15:20:35 -08:00
|
|
|
parent: psc,
|
|
|
|
spanContext: sc,
|
|
|
|
spanKind: trace.ValidateSpanKind(config.SpanKind()),
|
|
|
|
name: name,
|
|
|
|
startTime: startTime,
|
|
|
|
events: newEvictedQueue(tr.provider.spanLimits.EventCountLimit),
|
|
|
|
links: newEvictedQueue(tr.provider.spanLimits.LinkCountLimit),
|
|
|
|
tracer: tr,
|
2021-09-02 14:30:12 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, l := range config.Links() {
|
2024-03-28 16:35:15 +09:00
|
|
|
s.AddLink(l)
|
2021-09-02 14:30:12 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
s.SetAttributes(sr.Attributes...)
|
|
|
|
s.SetAttributes(config.Attributes()...)
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
2021-01-29 12:41:49 -08:00
|
|
|
|
2021-09-02 14:30:12 -07:00
|
|
|
// newNonRecordingSpan returns a new configured nonRecordingSpan.
|
|
|
|
func (tr *tracer) newNonRecordingSpan(sc trace.SpanContext) nonRecordingSpan {
|
|
|
|
return nonRecordingSpan{tracer: tr, sc: sc}
|
2019-08-02 13:52:55 -07:00
|
|
|
}
|