2020-03-23 22:41:10 -07:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2019-10-17 11:13:57 -07:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2020-10-12 08:47:41 -07:00
|
|
|
package oteltest
|
2019-10-17 11:13:57 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-10-24 16:05:26 -07:00
|
|
|
"sync"
|
2019-10-17 11:13:57 -07:00
|
|
|
"testing"
|
2019-10-24 16:05:26 -07:00
|
|
|
"time"
|
|
|
|
|
2020-10-08 19:58:56 -07:00
|
|
|
"go.opentelemetry.io/otel"
|
2020-08-10 09:17:09 -07:00
|
|
|
"go.opentelemetry.io/otel/codes"
|
2019-11-01 11:40:29 -07:00
|
|
|
"go.opentelemetry.io/otel/internal/matchers"
|
2020-08-17 20:25:03 -07:00
|
|
|
"go.opentelemetry.io/otel/label"
|
2019-10-17 11:13:57 -07:00
|
|
|
)
|
|
|
|
|
2020-10-12 08:47:41 -07:00
|
|
|
// Harness is a testing harness used to test implementations of the
|
|
|
|
// OpenTelemetry API.
|
2019-10-17 11:13:57 -07:00
|
|
|
type Harness struct {
|
|
|
|
t *testing.T
|
|
|
|
}
|
|
|
|
|
2020-10-12 08:47:41 -07:00
|
|
|
// NewHarness returns an instantiated *Harness using t.
|
2019-10-17 11:13:57 -07:00
|
|
|
func NewHarness(t *testing.T) *Harness {
|
|
|
|
return &Harness{
|
|
|
|
t: t,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-12 08:47:41 -07:00
|
|
|
// TestTracer runs validation tests for an implementation of the OpenTelemetry
|
|
|
|
// Tracer API.
|
2020-10-08 19:58:56 -07:00
|
|
|
func (h *Harness) TestTracer(subjectFactory func() otel.Tracer) {
|
2019-10-17 11:13:57 -07:00
|
|
|
h.t.Run("#Start", func(t *testing.T) {
|
|
|
|
t.Run("propagates the original context", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
ctxKey := testCtxKey{}
|
|
|
|
ctxValue := "ctx value"
|
|
|
|
ctx := context.WithValue(context.Background(), ctxKey, ctxValue)
|
|
|
|
|
|
|
|
ctx, _ = subject.Start(ctx, "test")
|
|
|
|
|
|
|
|
e.Expect(ctx.Value(ctxKey)).ToEqual(ctxValue)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("returns a span containing the expected properties", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
_, span := subject.Start(context.Background(), "test")
|
|
|
|
|
|
|
|
e.Expect(span).NotToBeNil()
|
|
|
|
|
|
|
|
e.Expect(span.Tracer()).ToEqual(subject)
|
|
|
|
e.Expect(span.SpanContext().IsValid()).ToBeTrue()
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("stores the span on the provided context", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
ctx, span := subject.Start(context.Background(), "test")
|
|
|
|
|
|
|
|
e.Expect(span).NotToBeNil()
|
2020-10-08 19:58:56 -07:00
|
|
|
e.Expect(span.SpanContext()).NotToEqual(otel.SpanContext{})
|
|
|
|
e.Expect(otel.SpanFromContext(ctx)).ToEqual(span)
|
2019-10-17 11:13:57 -07:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("starts spans with unique trace and span IDs", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
_, span1 := subject.Start(context.Background(), "span1")
|
|
|
|
_, span2 := subject.Start(context.Background(), "span2")
|
|
|
|
|
|
|
|
sc1 := span1.SpanContext()
|
|
|
|
sc2 := span2.SpanContext()
|
|
|
|
|
|
|
|
e.Expect(sc1.TraceID).NotToEqual(sc2.TraceID)
|
|
|
|
e.Expect(sc1.SpanID).NotToEqual(sc2.SpanID)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("records the span if specified", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
2020-10-08 19:58:56 -07:00
|
|
|
_, span := subject.Start(context.Background(), "span", otel.WithRecord())
|
2019-10-17 11:13:57 -07:00
|
|
|
|
|
|
|
e.Expect(span.IsRecording()).ToBeTrue()
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("propagates a parent's trace ID through the context", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
ctx, parent := subject.Start(context.Background(), "parent")
|
|
|
|
_, child := subject.Start(ctx, "child")
|
|
|
|
|
|
|
|
psc := parent.SpanContext()
|
|
|
|
csc := child.SpanContext()
|
|
|
|
|
|
|
|
e.Expect(csc.TraceID).ToEqual(psc.TraceID)
|
|
|
|
e.Expect(csc.SpanID).NotToEqual(psc.SpanID)
|
|
|
|
})
|
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
t.Run("ignores parent's trace ID when new root is requested", func(t *testing.T) {
|
2019-10-17 11:13:57 -07:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
ctx, parent := subject.Start(context.Background(), "parent")
|
2020-10-08 19:58:56 -07:00
|
|
|
_, child := subject.Start(ctx, "child", otel.WithNewRoot())
|
2019-10-17 11:13:57 -07:00
|
|
|
|
|
|
|
psc := parent.SpanContext()
|
|
|
|
csc := child.SpanContext()
|
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
e.Expect(csc.TraceID).NotToEqual(psc.TraceID)
|
2019-10-17 11:13:57 -07:00
|
|
|
e.Expect(csc.SpanID).NotToEqual(psc.SpanID)
|
|
|
|
})
|
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
t.Run("propagates remote parent's trace ID through the context", func(t *testing.T) {
|
2019-10-17 11:13:57 -07:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
_, remoteParent := subject.Start(context.Background(), "remote parent")
|
2020-10-08 19:58:56 -07:00
|
|
|
parentCtx := otel.ContextWithRemoteSpanContext(context.Background(), remoteParent.SpanContext())
|
2020-02-04 17:55:03 +01:00
|
|
|
_, child := subject.Start(parentCtx, "child")
|
2019-10-17 11:13:57 -07:00
|
|
|
|
2020-02-04 17:55:03 +01:00
|
|
|
psc := remoteParent.SpanContext()
|
2019-10-17 11:13:57 -07:00
|
|
|
csc := child.SpanContext()
|
|
|
|
|
|
|
|
e.Expect(csc.TraceID).ToEqual(psc.TraceID)
|
|
|
|
e.Expect(csc.SpanID).NotToEqual(psc.SpanID)
|
2020-02-04 17:55:03 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("ignores remote parent's trace ID when new root is requested", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
e := matchers.NewExpecter(t)
|
|
|
|
subject := subjectFactory()
|
|
|
|
|
|
|
|
_, remoteParent := subject.Start(context.Background(), "remote parent")
|
2020-10-08 19:58:56 -07:00
|
|
|
parentCtx := otel.ContextWithRemoteSpanContext(context.Background(), remoteParent.SpanContext())
|
|
|
|
_, child := subject.Start(parentCtx, "child", otel.WithNewRoot())
|
2020-02-04 17:55:03 +01:00
|
|
|
|
|
|
|
psc := remoteParent.SpanContext()
|
|
|
|
csc := child.SpanContext()
|
|
|
|
|
|
|
|
e.Expect(csc.TraceID).NotToEqual(psc.TraceID)
|
|
|
|
e.Expect(csc.SpanID).NotToEqual(psc.SpanID)
|
2019-10-17 11:13:57 -07:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-10-24 16:05:26 -07:00
|
|
|
h.testSpan(subjectFactory)
|
|
|
|
}
|
|
|
|
|
2020-10-08 19:58:56 -07:00
|
|
|
func (h *Harness) testSpan(tracerFactory func() otel.Tracer) {
|
|
|
|
var methods = map[string]func(span otel.Span){
|
|
|
|
"#End": func(span otel.Span) {
|
2019-10-24 16:05:26 -07:00
|
|
|
span.End()
|
|
|
|
},
|
2020-10-08 19:58:56 -07:00
|
|
|
"#AddEvent": func(span otel.Span) {
|
2019-10-24 16:05:26 -07:00
|
|
|
span.AddEvent(context.Background(), "test event")
|
|
|
|
},
|
2020-10-08 19:58:56 -07:00
|
|
|
"#AddEventWithTimestamp": func(span otel.Span) {
|
2019-10-24 16:05:26 -07:00
|
|
|
span.AddEventWithTimestamp(context.Background(), time.Now(), "test event")
|
|
|
|
},
|
2020-10-08 19:58:56 -07:00
|
|
|
"#SetStatus": func(span otel.Span) {
|
2020-10-05 11:36:03 -07:00
|
|
|
span.SetStatus(codes.Error, "internal")
|
2019-10-24 16:05:26 -07:00
|
|
|
},
|
2020-10-08 19:58:56 -07:00
|
|
|
"#SetName": func(span otel.Span) {
|
2019-10-24 16:05:26 -07:00
|
|
|
span.SetName("new name")
|
|
|
|
},
|
2020-10-08 19:58:56 -07:00
|
|
|
"#SetAttributes": func(span otel.Span) {
|
2020-08-17 20:25:03 -07:00
|
|
|
span.SetAttributes(label.String("key1", "value"), label.Int("key2", 123))
|
2019-10-24 16:05:26 -07:00
|
|
|
},
|
|
|
|
}
|
2020-10-08 19:58:56 -07:00
|
|
|
var mechanisms = map[string]func() otel.Span{
|
|
|
|
"Span created via Tracer#Start": func() otel.Span {
|
2019-10-24 16:05:26 -07:00
|
|
|
tracer := tracerFactory()
|
|
|
|
_, subject := tracer.Start(context.Background(), "test")
|
|
|
|
|
|
|
|
return subject
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for mechanismName, mechanism := range mechanisms {
|
|
|
|
h.t.Run(mechanismName, func(t *testing.T) {
|
|
|
|
for methodName, method := range methods {
|
|
|
|
t.Run(methodName, func(t *testing.T) {
|
|
|
|
t.Run("is thread-safe", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
span := mechanism()
|
|
|
|
|
|
|
|
wg := &sync.WaitGroup{}
|
|
|
|
wg.Add(2)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
method(span)
|
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
method(span)
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Run("#End", func(t *testing.T) {
|
|
|
|
t.Run("can be called multiple times", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
span := mechanism()
|
|
|
|
|
|
|
|
span.End()
|
|
|
|
span.End()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
2019-10-17 11:13:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
type testCtxKey struct{}
|