mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-26 03:52:03 +02:00
150 lines
4.0 KiB
Go
150 lines
4.0 KiB
Go
// Copyright The OpenTelemetry Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package oteltest // import "go.opentelemetry.io/otel/oteltest"
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"sync"
|
|
"sync/atomic"
|
|
|
|
"go.opentelemetry.io/otel"
|
|
)
|
|
|
|
// defaultSpanContextFunc returns the default SpanContextFunc.
|
|
func defaultSpanContextFunc() func(context.Context) otel.SpanContext {
|
|
var traceID, spanID uint64 = 1, 1
|
|
return func(ctx context.Context) otel.SpanContext {
|
|
var sc otel.SpanContext
|
|
if lsc := otel.SpanContextFromContext(ctx); lsc.IsValid() {
|
|
sc = lsc
|
|
} else if rsc := otel.RemoteSpanContextFromContext(ctx); rsc.IsValid() {
|
|
sc = rsc
|
|
} else {
|
|
binary.BigEndian.PutUint64(sc.TraceID[:], atomic.AddUint64(&traceID, 1))
|
|
}
|
|
binary.BigEndian.PutUint64(sc.SpanID[:], atomic.AddUint64(&spanID, 1))
|
|
return sc
|
|
}
|
|
}
|
|
|
|
type config struct {
|
|
// SpanContextFunc returns a SpanContext from an parent Context for a
|
|
// new span.
|
|
SpanContextFunc func(context.Context) otel.SpanContext
|
|
|
|
// SpanRecorder keeps track of spans.
|
|
SpanRecorder SpanRecorder
|
|
}
|
|
|
|
func newConfig(opts ...Option) config {
|
|
conf := config{}
|
|
for _, opt := range opts {
|
|
opt.Apply(&conf)
|
|
}
|
|
if conf.SpanContextFunc == nil {
|
|
conf.SpanContextFunc = defaultSpanContextFunc()
|
|
}
|
|
return conf
|
|
}
|
|
|
|
// Option applies an option to a config.
|
|
type Option interface {
|
|
Apply(*config)
|
|
}
|
|
|
|
type spanContextFuncOption struct {
|
|
SpanContextFunc func(context.Context) otel.SpanContext
|
|
}
|
|
|
|
func (o spanContextFuncOption) Apply(c *config) {
|
|
c.SpanContextFunc = o.SpanContextFunc
|
|
}
|
|
|
|
// WithSpanContextFunc sets the SpanContextFunc used to generate a new Spans
|
|
// context from a parent SpanContext.
|
|
func WithSpanContextFunc(f func(context.Context) otel.SpanContext) Option {
|
|
return spanContextFuncOption{f}
|
|
}
|
|
|
|
type spanRecorderOption struct {
|
|
SpanRecorder SpanRecorder
|
|
}
|
|
|
|
func (o spanRecorderOption) Apply(c *config) {
|
|
c.SpanRecorder = o.SpanRecorder
|
|
}
|
|
|
|
// WithSpanRecorder sets the SpanRecorder to use with the TracerProvider for
|
|
// testing.
|
|
func WithSpanRecorder(sr SpanRecorder) Option {
|
|
return spanRecorderOption{sr}
|
|
}
|
|
|
|
// SpanRecorder performs operations to record a span as it starts and ends.
|
|
type SpanRecorder interface {
|
|
// OnStart is called by the Tracer when it starts a Span.
|
|
OnStart(span *Span)
|
|
// OnEnd is called by the Span when it ends.
|
|
OnEnd(span *Span)
|
|
}
|
|
|
|
// StandardSpanRecorder is a SpanRecorder that records all started and ended
|
|
// spans in an ordered recording. StandardSpanRecorder is designed to be
|
|
// concurrent safe and can by used by multiple goroutines.
|
|
type StandardSpanRecorder struct {
|
|
startedMu sync.RWMutex
|
|
started []*Span
|
|
|
|
doneMu sync.RWMutex
|
|
done []*Span
|
|
}
|
|
|
|
// OnStart records span as started.
|
|
func (ssr *StandardSpanRecorder) OnStart(span *Span) {
|
|
ssr.startedMu.Lock()
|
|
defer ssr.startedMu.Unlock()
|
|
ssr.started = append(ssr.started, span)
|
|
}
|
|
|
|
// OnEnd records span as completed.
|
|
func (ssr *StandardSpanRecorder) OnEnd(span *Span) {
|
|
ssr.doneMu.Lock()
|
|
defer ssr.doneMu.Unlock()
|
|
ssr.done = append(ssr.done, span)
|
|
}
|
|
|
|
// Started returns a copy of all started Spans in the order they were started.
|
|
func (ssr *StandardSpanRecorder) Started() []*Span {
|
|
ssr.startedMu.RLock()
|
|
defer ssr.startedMu.RUnlock()
|
|
started := make([]*Span, len(ssr.started))
|
|
for i := range ssr.started {
|
|
started[i] = ssr.started[i]
|
|
}
|
|
return started
|
|
}
|
|
|
|
// Completed returns a copy of all ended Spans in the order they were ended.
|
|
func (ssr *StandardSpanRecorder) Completed() []*Span {
|
|
ssr.doneMu.RLock()
|
|
defer ssr.doneMu.RUnlock()
|
|
done := make([]*Span, len(ssr.done))
|
|
for i := range ssr.done {
|
|
done[i] = ssr.done[i]
|
|
}
|
|
return done
|
|
}
|