mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-12 02:28:07 +02:00
e53841a4b4
* Update Tracer API with instrumentation version Add option to the `Provider.Tracer` method to specify the instrumentation version. Update the global, noop, opentracing bridge, and default SDK implementations. This does not propagate the instrumentation library version to the exported span. That is left for a follow-on PR. * Revert trace_test.go This is for the next PR. * Support instrumentation library in SDK trace exports * Update Jaeger exporter to export instrumentation
1150 lines
32 KiB
Go
1150 lines
32 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 trace
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math"
|
|
"strings"
|
|
"sync/atomic"
|
|
"testing"
|
|
"time"
|
|
|
|
"go.opentelemetry.io/otel/api/global"
|
|
"go.opentelemetry.io/otel/api/kv/value"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"go.opentelemetry.io/otel/api/kv"
|
|
"go.opentelemetry.io/otel/api/testharness"
|
|
"go.opentelemetry.io/otel/api/trace"
|
|
apitrace "go.opentelemetry.io/otel/api/trace"
|
|
ottest "go.opentelemetry.io/otel/internal/testing"
|
|
export "go.opentelemetry.io/otel/sdk/export/trace"
|
|
"go.opentelemetry.io/otel/sdk/instrumentation"
|
|
"go.opentelemetry.io/otel/sdk/resource"
|
|
)
|
|
|
|
var (
|
|
tid apitrace.ID
|
|
sid apitrace.SpanID
|
|
)
|
|
|
|
type discardHandler struct{}
|
|
|
|
func (*discardHandler) Handle(_ error) {}
|
|
|
|
func init() {
|
|
tid, _ = apitrace.IDFromHex("01020304050607080102040810203040")
|
|
sid, _ = apitrace.SpanIDFromHex("0102040810203040")
|
|
|
|
global.SetHandler(new(discardHandler))
|
|
}
|
|
|
|
func TestTracerFollowsExpectedAPIBehaviour(t *testing.T) {
|
|
tp, err := NewProvider(WithConfig(Config{DefaultSampler: ProbabilitySampler(0)}))
|
|
if err != nil {
|
|
t.Fatalf("failed to create provider, err: %v\n", err)
|
|
}
|
|
harness := testharness.NewHarness(t)
|
|
subjectFactory := func() trace.Tracer {
|
|
return tp.Tracer("")
|
|
}
|
|
|
|
harness.TestTracer(subjectFactory)
|
|
}
|
|
|
|
type testExporter struct {
|
|
spans []*export.SpanData
|
|
}
|
|
|
|
func (t *testExporter) ExportSpan(ctx context.Context, d *export.SpanData) {
|
|
t.spans = append(t.spans, d)
|
|
}
|
|
|
|
type testSampler struct {
|
|
callCount int
|
|
prefix string
|
|
t *testing.T
|
|
}
|
|
|
|
func (ts *testSampler) ShouldSample(p SamplingParameters) SamplingResult {
|
|
ts.callCount++
|
|
ts.t.Logf("called sampler for name %q", p.Name)
|
|
decision := NotRecord
|
|
if strings.HasPrefix(p.Name, ts.prefix) {
|
|
decision = RecordAndSampled
|
|
}
|
|
return SamplingResult{Decision: decision, Attributes: []kv.KeyValue{kv.Int("callCount", ts.callCount)}}
|
|
}
|
|
|
|
func (ts testSampler) Description() string {
|
|
return "testSampler"
|
|
}
|
|
|
|
func TestSetName(t *testing.T) {
|
|
fooSampler := &testSampler{prefix: "foo", t: t}
|
|
tp, _ := NewProvider(WithConfig(Config{DefaultSampler: fooSampler}))
|
|
|
|
type testCase struct {
|
|
name string
|
|
newName string
|
|
sampledBefore bool
|
|
sampledAfter bool
|
|
}
|
|
for idx, tt := range []testCase{
|
|
{ // 0
|
|
name: "foobar",
|
|
newName: "foobaz",
|
|
sampledBefore: true,
|
|
sampledAfter: true,
|
|
},
|
|
{ // 1
|
|
name: "foobar",
|
|
newName: "barbaz",
|
|
sampledBefore: true,
|
|
sampledAfter: false,
|
|
},
|
|
{ // 2
|
|
name: "barbar",
|
|
newName: "barbaz",
|
|
sampledBefore: false,
|
|
sampledAfter: false,
|
|
},
|
|
{ // 3
|
|
name: "barbar",
|
|
newName: "foobar",
|
|
sampledBefore: false,
|
|
sampledAfter: true,
|
|
},
|
|
} {
|
|
span := startNamedSpan(tp, "SetName", tt.name)
|
|
if fooSampler.callCount == 0 {
|
|
t.Errorf("%d: the sampler was not even called during span creation", idx)
|
|
}
|
|
fooSampler.callCount = 0
|
|
if gotSampledBefore := span.SpanContext().IsSampled(); tt.sampledBefore != gotSampledBefore {
|
|
t.Errorf("%d: invalid sampling decision before rename, expected %v, got %v", idx, tt.sampledBefore, gotSampledBefore)
|
|
}
|
|
span.SetName(tt.newName)
|
|
if fooSampler.callCount == 0 {
|
|
t.Errorf("%d: the sampler was not even called during span rename", idx)
|
|
}
|
|
fooSampler.callCount = 0
|
|
if gotSampledAfter := span.SpanContext().IsSampled(); tt.sampledAfter != gotSampledAfter {
|
|
t.Errorf("%d: invalid sampling decision after rename, expected %v, got %v", idx, tt.sampledAfter, gotSampledAfter)
|
|
}
|
|
span.End()
|
|
}
|
|
}
|
|
|
|
func TestRecordingIsOn(t *testing.T) {
|
|
tp, _ := NewProvider()
|
|
_, span := tp.Tracer("Recording on").Start(context.Background(), "StartSpan")
|
|
defer span.End()
|
|
if span.IsRecording() == false {
|
|
t.Error("new span is not recording events")
|
|
}
|
|
}
|
|
|
|
func TestSampling(t *testing.T) {
|
|
idg := defIDGenerator()
|
|
const total = 10000
|
|
for name, tc := range map[string]struct {
|
|
sampler Sampler
|
|
expect float64
|
|
parent bool
|
|
sampledParent bool
|
|
}{
|
|
// Span w/o a parent
|
|
"NeverSample": {sampler: NeverSample(), expect: 0},
|
|
"AlwaysSample": {sampler: AlwaysSample(), expect: 1.0},
|
|
"ProbabilitySampler_-1": {sampler: ProbabilitySampler(-1.0), expect: 0},
|
|
"ProbabilitySampler_.25": {sampler: ProbabilitySampler(0.25), expect: .25},
|
|
"ProbabilitySampler_.50": {sampler: ProbabilitySampler(0.50), expect: .5},
|
|
"ProbabilitySampler_.75": {sampler: ProbabilitySampler(0.75), expect: .75},
|
|
"ProbabilitySampler_2.0": {sampler: ProbabilitySampler(2.0), expect: 1},
|
|
// Spans with a parent that is *not* sampled act like spans w/o a parent
|
|
"UnsampledParentSpanWithProbabilitySampler_-1": {sampler: ProbabilitySampler(-1.0), expect: 0, parent: true},
|
|
"UnsampledParentSpanWithProbabilitySampler_.25": {sampler: ProbabilitySampler(.25), expect: .25, parent: true},
|
|
"UnsampledParentSpanWithProbabilitySampler_.50": {sampler: ProbabilitySampler(0.50), expect: .5, parent: true},
|
|
"UnsampledParentSpanWithProbabilitySampler_.75": {sampler: ProbabilitySampler(0.75), expect: .75, parent: true},
|
|
"UnsampledParentSpanWithProbabilitySampler_2.0": {sampler: ProbabilitySampler(2.0), expect: 1, parent: true},
|
|
// Spans with a parent that is sampled, will always sample, regardless of the probability
|
|
"SampledParentSpanWithProbabilitySampler_-1": {sampler: ProbabilitySampler(-1.0), expect: 1, parent: true, sampledParent: true},
|
|
"SampledParentSpanWithProbabilitySampler_.25": {sampler: ProbabilitySampler(.25), expect: 1, parent: true, sampledParent: true},
|
|
"SampledParentSpanWithProbabilitySampler_2.0": {sampler: ProbabilitySampler(2.0), expect: 1, parent: true, sampledParent: true},
|
|
// Spans with a sampled parent, but when using the NeverSample Sampler, aren't sampled
|
|
"SampledParentSpanWithNeverSample": {sampler: NeverSample(), expect: 0, parent: true, sampledParent: true},
|
|
} {
|
|
tc := tc
|
|
t.Run(name, func(t *testing.T) {
|
|
t.Parallel()
|
|
p, err := NewProvider(WithConfig(Config{DefaultSampler: tc.sampler}))
|
|
if err != nil {
|
|
t.Fatal("unexpected error:", err)
|
|
}
|
|
tr := p.Tracer("test")
|
|
var sampled int
|
|
for i := 0; i < total; i++ {
|
|
ctx := context.Background()
|
|
if tc.parent {
|
|
psc := apitrace.SpanContext{
|
|
TraceID: idg.NewTraceID(),
|
|
SpanID: idg.NewSpanID(),
|
|
}
|
|
if tc.sampledParent {
|
|
psc.TraceFlags = apitrace.FlagsSampled
|
|
}
|
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx, psc)
|
|
}
|
|
_, span := tr.Start(ctx, "test")
|
|
if span.SpanContext().IsSampled() {
|
|
sampled++
|
|
}
|
|
}
|
|
tolerance := 0.0
|
|
got := float64(sampled) / float64(total)
|
|
|
|
if tc.expect > 0 && tc.expect < 1 {
|
|
// See https://en.wikipedia.org/wiki/Binomial_proportion_confidence_interval
|
|
const z = 4.75342 // This should succeed 99.9999% of the time
|
|
tolerance = z * math.Sqrt(got*(1-got)/total)
|
|
}
|
|
|
|
diff := math.Abs(got - tc.expect)
|
|
if diff > tolerance {
|
|
t.Errorf("got %f (diff: %f), expected %f (w/tolerance: %f)", got, diff, tc.expect, tolerance)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStartSpanWithParent(t *testing.T) {
|
|
tp, _ := NewProvider()
|
|
tr := tp.Tracer("SpanWithParent")
|
|
ctx := context.Background()
|
|
|
|
sc1 := apitrace.SpanContext{
|
|
TraceID: tid,
|
|
SpanID: sid,
|
|
TraceFlags: 0x1,
|
|
}
|
|
_, s1 := tr.Start(apitrace.ContextWithRemoteSpanContext(ctx, sc1), "span1-unsampled-parent1")
|
|
if err := checkChild(sc1, s1); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
_, s2 := tr.Start(apitrace.ContextWithRemoteSpanContext(ctx, sc1), "span2-unsampled-parent1")
|
|
if err := checkChild(sc1, s2); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
sc2 := apitrace.SpanContext{
|
|
TraceID: tid,
|
|
SpanID: sid,
|
|
TraceFlags: 0x1,
|
|
//Tracestate: testTracestate,
|
|
}
|
|
_, s3 := tr.Start(apitrace.ContextWithRemoteSpanContext(ctx, sc2), "span3-sampled-parent2")
|
|
if err := checkChild(sc2, s3); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
ctx2, s4 := tr.Start(apitrace.ContextWithRemoteSpanContext(ctx, sc2), "span4-sampled-parent2")
|
|
if err := checkChild(sc2, s4); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
s4Sc := s4.SpanContext()
|
|
_, s5 := tr.Start(ctx2, "span5-implicit-childof-span4")
|
|
if err := checkChild(s4Sc, s5); err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestSetSpanAttributesOnStart(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
span := startSpan(tp,
|
|
"StartSpanAttribute",
|
|
apitrace.WithAttributes(kv.String("key1", "value1")),
|
|
apitrace.WithAttributes(kv.String("key2", "value2")),
|
|
)
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
Attributes: []kv.KeyValue{
|
|
kv.String("key1", "value1"),
|
|
kv.String("key2", "value2"),
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "StartSpanAttribute"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SetSpanAttributesOnStart: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestSetSpanAttributes(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
span := startSpan(tp, "SpanAttribute")
|
|
span.SetAttributes(kv.Key("key1").String("value1"))
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
Attributes: []kv.KeyValue{
|
|
kv.String("key1", "value1"),
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "SpanAttribute"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SetSpanAttributes: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestSetSpanAttributesOverLimit(t *testing.T) {
|
|
te := &testExporter{}
|
|
cfg := Config{MaxAttributesPerSpan: 2}
|
|
tp, _ := NewProvider(WithConfig(cfg), WithSyncer(te))
|
|
|
|
span := startSpan(tp, "SpanAttributesOverLimit")
|
|
span.SetAttributes(
|
|
kv.Bool("key1", true),
|
|
kv.String("key2", "value2"),
|
|
kv.Bool("key1", false), // Replace key1.
|
|
kv.Int64("key4", 4), // Remove key2 and add key4
|
|
)
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
Attributes: []kv.KeyValue{
|
|
kv.Bool("key1", false),
|
|
kv.Int64("key4", 4),
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
DroppedAttributeCount: 1,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "SpanAttributesOverLimit"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SetSpanAttributesOverLimit: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestEvents(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
|
|
span := startSpan(tp, "Events")
|
|
k1v1 := kv.Key("key1").String("value1")
|
|
k2v2 := kv.Bool("key2", true)
|
|
k3v3 := kv.Int64("key3", 3)
|
|
|
|
span.AddEvent(context.Background(), "foo", kv.Key("key1").String("value1"))
|
|
span.AddEvent(context.Background(), "bar",
|
|
kv.Bool("key2", true),
|
|
kv.Int64("key3", 3),
|
|
)
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
for i := range got.MessageEvents {
|
|
if !checkTime(&got.MessageEvents[i].Time) {
|
|
t.Error("exporting span: expected nonzero Event Time")
|
|
}
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
HasRemoteParent: true,
|
|
MessageEvents: []export.Event{
|
|
{Name: "foo", Attributes: []kv.KeyValue{k1v1}},
|
|
{Name: "bar", Attributes: []kv.KeyValue{k2v2, k3v3}},
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "Events"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("Message Events: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestEventsOverLimit(t *testing.T) {
|
|
te := &testExporter{}
|
|
cfg := Config{MaxEventsPerSpan: 2}
|
|
tp, _ := NewProvider(WithConfig(cfg), WithSyncer(te))
|
|
|
|
span := startSpan(tp, "EventsOverLimit")
|
|
k1v1 := kv.Key("key1").String("value1")
|
|
k2v2 := kv.Bool("key2", false)
|
|
k3v3 := kv.Key("key3").String("value3")
|
|
|
|
span.AddEvent(context.Background(), "fooDrop", kv.Key("key1").String("value1"))
|
|
span.AddEvent(context.Background(), "barDrop",
|
|
kv.Bool("key2", true),
|
|
kv.Key("key3").String("value3"),
|
|
)
|
|
span.AddEvent(context.Background(), "foo", kv.Key("key1").String("value1"))
|
|
span.AddEvent(context.Background(), "bar",
|
|
kv.Bool("key2", false),
|
|
kv.Key("key3").String("value3"),
|
|
)
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
for i := range got.MessageEvents {
|
|
if !checkTime(&got.MessageEvents[i].Time) {
|
|
t.Error("exporting span: expected nonzero Event Time")
|
|
}
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
MessageEvents: []export.Event{
|
|
{Name: "foo", Attributes: []kv.KeyValue{k1v1}},
|
|
{Name: "bar", Attributes: []kv.KeyValue{k2v2, k3v3}},
|
|
},
|
|
DroppedMessageEventCount: 2,
|
|
HasRemoteParent: true,
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "EventsOverLimit"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("Message Event over limit: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestLinks(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
|
|
k1v1 := kv.Key("key1").String("value1")
|
|
k2v2 := kv.Key("key2").String("value2")
|
|
k3v3 := kv.Key("key3").String("value3")
|
|
|
|
sc1 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
|
sc2 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
|
|
|
span := startSpan(tp, "Links",
|
|
apitrace.LinkedTo(sc1, kv.Key("key1").String("value1")),
|
|
apitrace.LinkedTo(sc2,
|
|
kv.Key("key2").String("value2"),
|
|
kv.Key("key3").String("value3"),
|
|
),
|
|
)
|
|
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
HasRemoteParent: true,
|
|
Links: []apitrace.Link{
|
|
{SpanContext: sc1, Attributes: []kv.KeyValue{k1v1}},
|
|
{SpanContext: sc2, Attributes: []kv.KeyValue{k2v2, k3v3}},
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "Links"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("Link: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestLinksOverLimit(t *testing.T) {
|
|
te := &testExporter{}
|
|
cfg := Config{MaxLinksPerSpan: 2}
|
|
|
|
sc1 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
|
sc2 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
|
sc3 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
|
|
|
tp, _ := NewProvider(WithConfig(cfg), WithSyncer(te))
|
|
|
|
span := startSpan(tp, "LinksOverLimit",
|
|
apitrace.LinkedTo(sc1, kv.Key("key1").String("value1")),
|
|
apitrace.LinkedTo(sc2, kv.Key("key2").String("value2")),
|
|
apitrace.LinkedTo(sc3, kv.Key("key3").String("value3")),
|
|
)
|
|
|
|
k2v2 := kv.Key("key2").String("value2")
|
|
k3v3 := kv.Key("key3").String("value3")
|
|
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
Links: []apitrace.Link{
|
|
{SpanContext: sc2, Attributes: []kv.KeyValue{k2v2}},
|
|
{SpanContext: sc3, Attributes: []kv.KeyValue{k3v3}},
|
|
},
|
|
DroppedLinkCount: 1,
|
|
HasRemoteParent: true,
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "LinksOverLimit"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("Link over limit: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestSetSpanName(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
ctx := context.Background()
|
|
|
|
want := "SpanName-1"
|
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx, apitrace.SpanContext{
|
|
TraceID: tid,
|
|
SpanID: sid,
|
|
TraceFlags: 1,
|
|
})
|
|
_, span := tp.Tracer("SetSpanName").Start(ctx, "SpanName-1")
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if got.Name != want {
|
|
t.Errorf("span.Name: got %q; want %q", got.Name, want)
|
|
}
|
|
}
|
|
|
|
func TestSetSpanStatus(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
|
|
span := startSpan(tp, "SpanStatus")
|
|
span.SetStatus(codes.Canceled, "canceled")
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
StatusCode: codes.Canceled,
|
|
StatusMessage: "canceled",
|
|
HasRemoteParent: true,
|
|
InstrumentationLibrary: instrumentation.Library{Name: "SpanStatus"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SetSpanStatus: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func cmpDiff(x, y interface{}) string {
|
|
return cmp.Diff(x, y,
|
|
cmp.AllowUnexported(value.Value{}),
|
|
cmp.AllowUnexported(export.Event{}))
|
|
}
|
|
|
|
func remoteSpanContext() apitrace.SpanContext {
|
|
return apitrace.SpanContext{
|
|
TraceID: tid,
|
|
SpanID: sid,
|
|
TraceFlags: 1,
|
|
}
|
|
}
|
|
|
|
// checkChild is test utility function that tests that c has fields set appropriately,
|
|
// given that it is a child span of p.
|
|
func checkChild(p apitrace.SpanContext, apiSpan apitrace.Span) error {
|
|
s := apiSpan.(*span)
|
|
if s == nil {
|
|
return fmt.Errorf("got nil child span, want non-nil")
|
|
}
|
|
if got, want := s.spanContext.TraceID.String(), p.TraceID.String(); got != want {
|
|
return fmt.Errorf("got child trace ID %s, want %s", got, want)
|
|
}
|
|
if childID, parentID := s.spanContext.SpanID.String(), p.SpanID.String(); childID == parentID {
|
|
return fmt.Errorf("got child span ID %s, parent span ID %s; want unequal IDs", childID, parentID)
|
|
}
|
|
if got, want := s.spanContext.TraceFlags, p.TraceFlags; got != want {
|
|
return fmt.Errorf("got child trace options %d, want %d", got, want)
|
|
}
|
|
// TODO [rgheita] : Fix tracestate test
|
|
//if got, want := c.spanContext.Tracestate, p.Tracestate; got != want {
|
|
// return fmt.Errorf("got child tracestate %v, want %v", got, want)
|
|
//}
|
|
return nil
|
|
}
|
|
|
|
// startSpan starts a span with a name "span0". See startNamedSpan for
|
|
// details.
|
|
func startSpan(tp *Provider, trName string, args ...apitrace.StartOption) apitrace.Span {
|
|
return startNamedSpan(tp, trName, "span0", args...)
|
|
}
|
|
|
|
// startNamed Span is a test utility func that starts a span with a
|
|
// passed name and with remote span context as parent. The remote span
|
|
// context contains TraceFlags with sampled bit set. This allows the
|
|
// span to be automatically sampled.
|
|
func startNamedSpan(tp *Provider, trName, name string, args ...apitrace.StartOption) apitrace.Span {
|
|
ctx := context.Background()
|
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx, remoteSpanContext())
|
|
args = append(args, apitrace.WithRecord())
|
|
_, span := tp.Tracer(trName).Start(
|
|
ctx,
|
|
name,
|
|
args...,
|
|
)
|
|
return span
|
|
}
|
|
|
|
// endSpan is a test utility function that ends the span in the context and
|
|
// returns the exported export.SpanData.
|
|
// It requires that span be sampled using one of these methods
|
|
// 1. Passing parent span context in context
|
|
// 2. Use WithSampler(AlwaysSample())
|
|
// 3. Configuring AlwaysSample() as default sampler
|
|
//
|
|
// It also does some basic tests on the span.
|
|
// It also clears spanID in the export.SpanData to make the comparison easier.
|
|
func endSpan(te *testExporter, span apitrace.Span) (*export.SpanData, error) {
|
|
if !span.IsRecording() {
|
|
return nil, fmt.Errorf("IsRecording: got false, want true")
|
|
}
|
|
if !span.SpanContext().IsSampled() {
|
|
return nil, fmt.Errorf("IsSampled: got false, want true")
|
|
}
|
|
span.End()
|
|
if len(te.spans) != 1 {
|
|
return nil, fmt.Errorf("got exported spans %#v, want one span", te.spans)
|
|
}
|
|
got := te.spans[0]
|
|
if !got.SpanContext.SpanID.IsValid() {
|
|
return nil, fmt.Errorf("exporting span: expected nonzero SpanID")
|
|
}
|
|
got.SpanContext.SpanID = apitrace.SpanID{}
|
|
if !checkTime(&got.StartTime) {
|
|
return nil, fmt.Errorf("exporting span: expected nonzero StartTime")
|
|
}
|
|
if !checkTime(&got.EndTime) {
|
|
return nil, fmt.Errorf("exporting span: expected nonzero EndTime")
|
|
}
|
|
return got, nil
|
|
}
|
|
|
|
// checkTime checks that a nonzero time was set in x, then clears it.
|
|
func checkTime(x *time.Time) bool {
|
|
if x.IsZero() {
|
|
return false
|
|
}
|
|
*x = time.Time{}
|
|
return true
|
|
}
|
|
|
|
type fakeExporter map[string]*export.SpanData
|
|
|
|
func (f fakeExporter) ExportSpan(ctx context.Context, s *export.SpanData) {
|
|
f[s.Name] = s
|
|
}
|
|
|
|
func TestEndSpanTwice(t *testing.T) {
|
|
spans := make(fakeExporter)
|
|
tp, _ := NewProvider(WithSyncer(spans))
|
|
|
|
span := startSpan(tp, "EndSpanTwice")
|
|
span.End()
|
|
span.End()
|
|
if len(spans) != 1 {
|
|
t.Fatalf("expected only a single span, got %#v", spans)
|
|
}
|
|
}
|
|
|
|
func TestStartSpanAfterEnd(t *testing.T) {
|
|
spans := make(fakeExporter)
|
|
tp, _ := NewProvider(WithConfig(Config{DefaultSampler: AlwaysSample()}), WithSyncer(spans))
|
|
ctx := context.Background()
|
|
|
|
tr := tp.Tracer("SpanAfterEnd")
|
|
ctx, span0 := tr.Start(apitrace.ContextWithRemoteSpanContext(ctx, remoteSpanContext()), "parent")
|
|
ctx1, span1 := tr.Start(ctx, "span-1")
|
|
span1.End()
|
|
// Start a new span with the context containing span-1
|
|
// even though span-1 is ended, we still add this as a new child of span-1
|
|
_, span2 := tr.Start(ctx1, "span-2")
|
|
span2.End()
|
|
span0.End()
|
|
if got, want := len(spans), 3; got != want {
|
|
t.Fatalf("len(%#v) = %d; want %d", spans, got, want)
|
|
}
|
|
|
|
gotParent, ok := spans["parent"]
|
|
if !ok {
|
|
t.Fatal("parent not recorded")
|
|
}
|
|
gotSpan1, ok := spans["span-1"]
|
|
if !ok {
|
|
t.Fatal("span-1 not recorded")
|
|
}
|
|
gotSpan2, ok := spans["span-2"]
|
|
if !ok {
|
|
t.Fatal("span-2 not recorded")
|
|
}
|
|
|
|
if got, want := gotSpan1.SpanContext.TraceID, gotParent.SpanContext.TraceID; got != want {
|
|
t.Errorf("span-1.TraceID=%q; want %q", got, want)
|
|
}
|
|
if got, want := gotSpan2.SpanContext.TraceID, gotParent.SpanContext.TraceID; got != want {
|
|
t.Errorf("span-2.TraceID=%q; want %q", got, want)
|
|
}
|
|
if got, want := gotSpan1.ParentSpanID, gotParent.SpanContext.SpanID; got != want {
|
|
t.Errorf("span-1.ParentSpanID=%q; want %q (parent.SpanID)", got, want)
|
|
}
|
|
if got, want := gotSpan2.ParentSpanID, gotSpan1.SpanContext.SpanID; got != want {
|
|
t.Errorf("span-2.ParentSpanID=%q; want %q (span1.SpanID)", got, want)
|
|
}
|
|
}
|
|
|
|
func TestChildSpanCount(t *testing.T) {
|
|
spans := make(fakeExporter)
|
|
tp, _ := NewProvider(WithConfig(Config{DefaultSampler: AlwaysSample()}), WithSyncer(spans))
|
|
|
|
tr := tp.Tracer("ChidSpanCount")
|
|
ctx, span0 := tr.Start(context.Background(), "parent")
|
|
ctx1, span1 := tr.Start(ctx, "span-1")
|
|
_, span2 := tr.Start(ctx1, "span-2")
|
|
span2.End()
|
|
span1.End()
|
|
|
|
_, span3 := tr.Start(ctx, "span-3")
|
|
span3.End()
|
|
span0.End()
|
|
if got, want := len(spans), 4; got != want {
|
|
t.Fatalf("len(%#v) = %d; want %d", spans, got, want)
|
|
}
|
|
|
|
gotParent, ok := spans["parent"]
|
|
if !ok {
|
|
t.Fatal("parent not recorded")
|
|
}
|
|
gotSpan1, ok := spans["span-1"]
|
|
if !ok {
|
|
t.Fatal("span-1 not recorded")
|
|
}
|
|
gotSpan2, ok := spans["span-2"]
|
|
if !ok {
|
|
t.Fatal("span-2 not recorded")
|
|
}
|
|
gotSpan3, ok := spans["span-3"]
|
|
if !ok {
|
|
t.Fatal("span-3 not recorded")
|
|
}
|
|
|
|
if got, want := gotSpan3.ChildSpanCount, 0; got != want {
|
|
t.Errorf("span-3.ChildSpanCount=%q; want %q", got, want)
|
|
}
|
|
if got, want := gotSpan2.ChildSpanCount, 0; got != want {
|
|
t.Errorf("span-2.ChildSpanCount=%q; want %q", got, want)
|
|
}
|
|
if got, want := gotSpan1.ChildSpanCount, 1; got != want {
|
|
t.Errorf("span-1.ChildSpanCount=%q; want %q", got, want)
|
|
}
|
|
if got, want := gotParent.ChildSpanCount, 2; got != want {
|
|
t.Errorf("parent.ChildSpanCount=%q; want %q", got, want)
|
|
}
|
|
}
|
|
|
|
func TestNilSpanEnd(t *testing.T) {
|
|
var span *span
|
|
span.End()
|
|
}
|
|
|
|
func TestExecutionTracerTaskEnd(t *testing.T) {
|
|
var n uint64
|
|
tp, _ := NewProvider(WithConfig(Config{DefaultSampler: NeverSample()}))
|
|
tr := tp.Tracer("Execution Tracer Task End")
|
|
|
|
executionTracerTaskEnd := func() {
|
|
atomic.AddUint64(&n, 1)
|
|
}
|
|
|
|
var spans []*span
|
|
_, apiSpan := tr.Start(context.Background(), "foo")
|
|
s := apiSpan.(*span)
|
|
|
|
s.executionTracerTaskEnd = executionTracerTaskEnd
|
|
spans = append(spans, s) // never sample
|
|
|
|
tID, _ := apitrace.IDFromHex("0102030405060708090a0b0c0d0e0f")
|
|
sID, _ := apitrace.SpanIDFromHex("0001020304050607")
|
|
ctx := context.Background()
|
|
|
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx,
|
|
apitrace.SpanContext{
|
|
TraceID: tID,
|
|
SpanID: sID,
|
|
TraceFlags: 0,
|
|
},
|
|
)
|
|
_, apiSpan = tr.Start(
|
|
ctx,
|
|
"foo",
|
|
)
|
|
s = apiSpan.(*span)
|
|
s.executionTracerTaskEnd = executionTracerTaskEnd
|
|
spans = append(spans, s) // parent not sampled
|
|
|
|
//tp.ApplyConfig(Config{DefaultSampler: AlwaysSample()})
|
|
_, apiSpan = tr.Start(context.Background(), "foo")
|
|
s = apiSpan.(*span)
|
|
s.executionTracerTaskEnd = executionTracerTaskEnd
|
|
spans = append(spans, s) // always sample
|
|
|
|
for _, span := range spans {
|
|
span.End()
|
|
}
|
|
if got, want := n, uint64(len(spans)); got != want {
|
|
t.Fatalf("Execution tracer task ended for %v spans; want %v", got, want)
|
|
}
|
|
}
|
|
|
|
func TestCustomStartEndTime(t *testing.T) {
|
|
var te testExporter
|
|
tp, _ := NewProvider(WithSyncer(&te), WithConfig(Config{DefaultSampler: AlwaysSample()}))
|
|
|
|
startTime := time.Date(2019, time.August, 27, 14, 42, 0, 0, time.UTC)
|
|
endTime := startTime.Add(time.Second * 20)
|
|
_, span := tp.Tracer("Custom Start and End time").Start(
|
|
context.Background(),
|
|
"testspan",
|
|
apitrace.WithStartTime(startTime),
|
|
)
|
|
span.End(apitrace.WithEndTime(endTime))
|
|
|
|
if len(te.spans) != 1 {
|
|
t.Fatalf("got exported spans %#v, want one span", te.spans)
|
|
}
|
|
got := te.spans[0]
|
|
if got.StartTime != startTime {
|
|
t.Errorf("expected start time to be %s, got %s", startTime, got.StartTime)
|
|
}
|
|
if got.EndTime != endTime {
|
|
t.Errorf("expected end time to be %s, got %s", endTime, got.EndTime)
|
|
}
|
|
}
|
|
|
|
func TestRecordError(t *testing.T) {
|
|
scenarios := []struct {
|
|
err error
|
|
typ string
|
|
msg string
|
|
}{
|
|
{
|
|
err: ottest.NewTestError("test error"),
|
|
typ: "go.opentelemetry.io/otel/internal/testing.TestError",
|
|
msg: "test error",
|
|
},
|
|
{
|
|
err: errors.New("test error 2"),
|
|
typ: "*errors.errorString",
|
|
msg: "test error 2",
|
|
},
|
|
}
|
|
|
|
for _, s := range scenarios {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
span := startSpan(tp, "RecordError")
|
|
|
|
errTime := time.Now()
|
|
span.RecordError(context.Background(), s.err,
|
|
apitrace.WithErrorTime(errTime),
|
|
)
|
|
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
MessageEvents: []export.Event{
|
|
{
|
|
Name: errorEventName,
|
|
Time: errTime,
|
|
Attributes: []kv.KeyValue{
|
|
errorTypeKey.String(s.typ),
|
|
errorMessageKey.String(s.msg),
|
|
},
|
|
},
|
|
},
|
|
InstrumentationLibrary: instrumentation.Library{Name: "RecordError"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SpanErrorOptions: -got +want %s", diff)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestRecordErrorWithStatus(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
span := startSpan(tp, "RecordErrorWithStatus")
|
|
|
|
testErr := ottest.NewTestError("test error")
|
|
errTime := time.Now()
|
|
testStatus := codes.Unknown
|
|
span.RecordError(context.Background(), testErr,
|
|
apitrace.WithErrorTime(errTime),
|
|
apitrace.WithErrorStatus(testStatus),
|
|
)
|
|
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
StatusCode: codes.Unknown,
|
|
StatusMessage: "",
|
|
HasRemoteParent: true,
|
|
MessageEvents: []export.Event{
|
|
{
|
|
Name: errorEventName,
|
|
Time: errTime,
|
|
Attributes: []kv.KeyValue{
|
|
errorTypeKey.String("go.opentelemetry.io/otel/internal/testing.TestError"),
|
|
errorMessageKey.String("test error"),
|
|
},
|
|
},
|
|
},
|
|
InstrumentationLibrary: instrumentation.Library{Name: "RecordErrorWithStatus"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SpanErrorOptions: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestRecordErrorNil(t *testing.T) {
|
|
te := &testExporter{}
|
|
tp, _ := NewProvider(WithSyncer(te))
|
|
span := startSpan(tp, "RecordErrorNil")
|
|
|
|
span.RecordError(context.Background(), nil)
|
|
|
|
got, err := endSpan(te, span)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
StatusCode: codes.OK,
|
|
StatusMessage: "",
|
|
InstrumentationLibrary: instrumentation.Library{Name: "RecordErrorNil"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("SpanErrorOptions: -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestWithSpanKind(t *testing.T) {
|
|
var te testExporter
|
|
tp, _ := NewProvider(WithSyncer(&te), WithConfig(Config{DefaultSampler: AlwaysSample()}))
|
|
tr := tp.Tracer("withSpanKind")
|
|
|
|
_, span := tr.Start(context.Background(), "WithoutSpanKind")
|
|
spanData, err := endSpan(&te, span)
|
|
if err != nil {
|
|
t.Error(err.Error())
|
|
}
|
|
|
|
if spanData.SpanKind != apitrace.SpanKindInternal {
|
|
t.Errorf("Default value of Spankind should be Internal: got %+v, want %+v\n", spanData.SpanKind, apitrace.SpanKindInternal)
|
|
}
|
|
|
|
sks := []apitrace.SpanKind{
|
|
apitrace.SpanKindInternal,
|
|
apitrace.SpanKindServer,
|
|
apitrace.SpanKindClient,
|
|
apitrace.SpanKindProducer,
|
|
apitrace.SpanKindConsumer,
|
|
}
|
|
|
|
for _, sk := range sks {
|
|
te.spans = nil
|
|
|
|
_, span := tr.Start(context.Background(), fmt.Sprintf("SpanKind-%v", sk), apitrace.WithSpanKind(sk))
|
|
spanData, err := endSpan(&te, span)
|
|
if err != nil {
|
|
t.Error(err.Error())
|
|
}
|
|
|
|
if spanData.SpanKind != sk {
|
|
t.Errorf("WithSpanKind check: got %+v, want %+v\n", spanData.SpanKind, sks)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestWithResource(t *testing.T) {
|
|
var te testExporter
|
|
tp, _ := NewProvider(WithSyncer(&te),
|
|
WithConfig(Config{DefaultSampler: AlwaysSample()}),
|
|
WithResource(resource.New(kv.String("rk1", "rv1"), kv.Int64("rk2", 5))))
|
|
span := startSpan(tp, "WithResource")
|
|
span.SetAttributes(kv.String("key1", "value1"))
|
|
got, err := endSpan(&te, span)
|
|
if err != nil {
|
|
t.Error(err.Error())
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
Attributes: []kv.KeyValue{
|
|
kv.String("key1", "value1"),
|
|
},
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
Resource: resource.New(kv.String("rk1", "rv1"), kv.Int64("rk2", 5)),
|
|
InstrumentationLibrary: instrumentation.Library{Name: "WithResource"},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("WithResource:\n -got +want %s", diff)
|
|
}
|
|
}
|
|
|
|
func TestWithInstrumentationVersion(t *testing.T) {
|
|
var te testExporter
|
|
tp, _ := NewProvider(WithSyncer(&te))
|
|
|
|
ctx := context.Background()
|
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx, remoteSpanContext())
|
|
_, span := tp.Tracer(
|
|
"WithInstrumentationVersion",
|
|
apitrace.WithInstrumentationVersion("v0.1.0"),
|
|
).Start(ctx, "span0", apitrace.WithRecord())
|
|
got, err := endSpan(&te, span)
|
|
if err != nil {
|
|
t.Error(err.Error())
|
|
}
|
|
|
|
want := &export.SpanData{
|
|
SpanContext: apitrace.SpanContext{
|
|
TraceID: tid,
|
|
TraceFlags: 0x1,
|
|
},
|
|
ParentSpanID: sid,
|
|
Name: "span0",
|
|
SpanKind: apitrace.SpanKindInternal,
|
|
HasRemoteParent: true,
|
|
InstrumentationLibrary: instrumentation.Library{
|
|
Name: "WithInstrumentationVersion",
|
|
Version: "v0.1.0",
|
|
},
|
|
}
|
|
if diff := cmpDiff(got, want); diff != "" {
|
|
t.Errorf("WithResource:\n -got +want %s", diff)
|
|
}
|
|
}
|