// 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/trace" ) // defaultSpanContextFunc returns the default SpanContextFunc. func defaultSpanContextFunc() func(context.Context) trace.SpanContext { var traceID, spanID uint64 = 1, 1 return func(ctx context.Context) trace.SpanContext { var sc trace.SpanContext if current := trace.SpanContextFromContext(ctx); current.IsValid() { sc = current } else { var tid trace.TraceID binary.BigEndian.PutUint64(tid[:], atomic.AddUint64(&traceID, 1)) sc = sc.WithTraceID(tid) } var sid trace.SpanID binary.BigEndian.PutUint64(sid[:], atomic.AddUint64(&spanID, 1)) return sc.WithSpanID(sid) } } type config struct { // SpanContextFunc returns a SpanContext from an parent Context for a // new span. SpanContextFunc func(context.Context) trace.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) trace.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) trace.SpanContext) Option { return spanContextFuncOption{SpanContextFunc: 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{SpanRecorder: sr} } // SpanRecorder performs operations to record a span as it starts and ends. // It is designed to be concurrent safe and can by used by multiple goroutines. type SpanRecorder struct { startedMu sync.RWMutex started []*Span doneMu sync.RWMutex done []*Span } // OnStart records span as started. func (ssr *SpanRecorder) OnStart(span *Span) { ssr.startedMu.Lock() defer ssr.startedMu.Unlock() ssr.started = append(ssr.started, span) } // OnEnd records span as completed. func (ssr *SpanRecorder) 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 *SpanRecorder) 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 *SpanRecorder) 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 }