2020-03-23 22:41:10 -07:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2019-11-03 20:46:57 -03: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.
|
|
|
|
package httptrace_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2020-04-07 12:22:50 -04:00
|
|
|
nhtrace "net/http/httptrace"
|
2019-11-03 20:46:57 -03:00
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
|
2019-11-26 12:41:24 +08:00
|
|
|
"go.opentelemetry.io/otel/api/global"
|
2020-05-14 07:06:03 +08:00
|
|
|
"go.opentelemetry.io/otel/api/kv"
|
2020-05-30 14:53:32 -07:00
|
|
|
"go.opentelemetry.io/otel/instrumentation/httptrace"
|
2019-11-05 13:08:55 -08:00
|
|
|
export "go.opentelemetry.io/otel/sdk/export/trace"
|
2019-11-03 20:46:57 -03:00
|
|
|
sdktrace "go.opentelemetry.io/otel/sdk/trace"
|
|
|
|
)
|
|
|
|
|
|
|
|
type testExporter struct {
|
|
|
|
mu sync.Mutex
|
|
|
|
spanMap map[string][]*export.SpanData
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *testExporter) ExportSpan(ctx context.Context, s *export.SpanData) {
|
|
|
|
t.mu.Lock()
|
|
|
|
defer t.mu.Unlock()
|
|
|
|
var spans []*export.SpanData
|
|
|
|
var ok bool
|
|
|
|
|
|
|
|
if spans, ok = t.spanMap[s.Name]; !ok {
|
|
|
|
spans = []*export.SpanData{}
|
|
|
|
t.spanMap[s.Name] = spans
|
|
|
|
}
|
|
|
|
spans = append(spans, s)
|
|
|
|
t.spanMap[s.Name] = spans
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ export.SpanSyncer = (*testExporter)(nil)
|
|
|
|
|
|
|
|
func TestHTTPRequestWithClientTrace(t *testing.T) {
|
|
|
|
exp := &testExporter{
|
|
|
|
spanMap: make(map[string][]*export.SpanData),
|
|
|
|
}
|
|
|
|
tp, _ := sdktrace.NewProvider(sdktrace.WithSyncer(exp), sdktrace.WithConfig(sdktrace.Config{DefaultSampler: sdktrace.AlwaysSample()}))
|
|
|
|
global.SetTraceProvider(tp)
|
|
|
|
|
2019-11-25 18:46:07 +01:00
|
|
|
tr := tp.Tracer("httptrace/client")
|
2019-11-03 20:46:57 -03:00
|
|
|
|
|
|
|
// Mock http server
|
|
|
|
ts := httptest.NewServer(
|
|
|
|
http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
address := ts.Listener.Addr()
|
|
|
|
|
|
|
|
client := ts.Client()
|
|
|
|
err := tr.WithSpan(context.Background(), "test",
|
|
|
|
func(ctx context.Context) error {
|
|
|
|
req, _ := http.NewRequest("GET", ts.URL, nil)
|
|
|
|
_, req = httptrace.W3C(ctx, req)
|
|
|
|
|
|
|
|
res, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Request failed: %s", err.Error())
|
|
|
|
}
|
|
|
|
_ = res.Body.Close()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic("unexpected error in http request: " + err.Error())
|
|
|
|
}
|
|
|
|
|
2020-04-07 12:22:50 -04:00
|
|
|
getSpan := func(name string) *export.SpanData {
|
|
|
|
spans, ok := exp.spanMap[name]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("no spans found with the name %s, %v", name, exp.spanMap)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(spans) != 1 {
|
|
|
|
t.Fatalf("Expected exactly one span for %s but found %d", name, len(spans))
|
|
|
|
}
|
|
|
|
|
|
|
|
return spans[0]
|
|
|
|
}
|
|
|
|
|
2019-11-03 20:46:57 -03:00
|
|
|
testLen := []struct {
|
|
|
|
name string
|
2020-05-14 07:06:03 +08:00
|
|
|
attributes []kv.KeyValue
|
2020-04-07 12:22:50 -04:00
|
|
|
parent string
|
2019-11-03 20:46:57 -03:00
|
|
|
}{
|
|
|
|
{
|
2020-01-14 04:54:48 -08:00
|
|
|
name: "http.connect",
|
2020-05-14 07:06:03 +08:00
|
|
|
attributes: []kv.KeyValue{kv.String("http.remote", address.String())},
|
2020-04-07 12:22:50 -04:00
|
|
|
parent: "http.getconn",
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
|
|
|
{
|
2020-01-14 04:54:48 -08:00
|
|
|
name: "http.getconn",
|
2020-05-14 07:06:03 +08:00
|
|
|
attributes: []kv.KeyValue{
|
|
|
|
kv.String("http.remote", address.String()),
|
|
|
|
kv.String("http.host", address.String()),
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
2020-04-07 12:22:50 -04:00
|
|
|
parent: "test",
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
|
|
|
{
|
2020-04-07 12:22:50 -04:00
|
|
|
name: "http.receive",
|
|
|
|
parent: "test",
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
|
|
|
{
|
2020-04-07 12:22:50 -04:00
|
|
|
name: "http.send",
|
|
|
|
parent: "test",
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
|
|
|
{
|
2020-01-14 04:54:48 -08:00
|
|
|
name: "test",
|
2019-11-03 20:46:57 -03:00
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tl := range testLen {
|
2020-04-07 12:22:50 -04:00
|
|
|
span := getSpan(tl.name)
|
2019-11-03 20:46:57 -03:00
|
|
|
|
2020-04-07 12:22:50 -04:00
|
|
|
if tl.parent != "" {
|
|
|
|
parentSpan := getSpan(tl.parent)
|
|
|
|
|
|
|
|
if span.ParentSpanID != parentSpan.SpanContext.SpanID {
|
|
|
|
t.Fatalf("[span %s] does not have expected parent span %s", tl.name, tl.parent)
|
|
|
|
}
|
2019-11-03 20:46:57 -03:00
|
|
|
}
|
|
|
|
|
2020-05-14 07:06:03 +08:00
|
|
|
actualAttrs := make(map[kv.Key]string)
|
2019-11-03 20:46:57 -03:00
|
|
|
for _, attr := range span.Attributes {
|
|
|
|
actualAttrs[attr.Key] = attr.Value.Emit()
|
|
|
|
}
|
|
|
|
|
2020-05-14 07:06:03 +08:00
|
|
|
expectedAttrs := make(map[kv.Key]string)
|
2019-11-03 20:46:57 -03:00
|
|
|
for _, attr := range tl.attributes {
|
|
|
|
expectedAttrs[attr.Key] = attr.Value.Emit()
|
|
|
|
}
|
|
|
|
|
2020-01-14 04:54:48 -08:00
|
|
|
if tl.name == "http.getconn" {
|
2020-05-13 16:21:23 -07:00
|
|
|
local := kv.Key("http.local")
|
2019-11-03 20:46:57 -03:00
|
|
|
// http.local attribute is not deterministic, just make sure it exists for `getconn`.
|
|
|
|
if _, ok := actualAttrs[local]; ok {
|
|
|
|
delete(actualAttrs, local)
|
|
|
|
} else {
|
|
|
|
t.Fatalf("[span %s] is missing attribute %v", tl.name, local)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if diff := cmp.Diff(actualAttrs, expectedAttrs); diff != "" {
|
|
|
|
t.Fatalf("[span %s] Attributes are different: %v", tl.name, diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestConcurrentConnectionStart(t *testing.T) {
|
|
|
|
exp := &testExporter{
|
|
|
|
spanMap: make(map[string][]*export.SpanData),
|
|
|
|
}
|
|
|
|
tp, _ := sdktrace.NewProvider(sdktrace.WithSyncer(exp), sdktrace.WithConfig(sdktrace.Config{DefaultSampler: sdktrace.AlwaysSample()}))
|
|
|
|
global.SetTraceProvider(tp)
|
|
|
|
|
|
|
|
ct := httptrace.NewClientTrace(context.Background())
|
|
|
|
|
|
|
|
tts := []struct {
|
|
|
|
name string
|
|
|
|
run func()
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "Open1Close1Open2Close2",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Open2Close2Open1Close1",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Open1Open2Close1Close2",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Open1Open2Close2Close1",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Open2Open1Close1Close2",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Open2Open1Close2Close1",
|
|
|
|
run: func() {
|
|
|
|
exp.spanMap = make(map[string][]*export.SpanData)
|
|
|
|
|
|
|
|
ct.ConnectStart("tcp", "[::1]:3000")
|
|
|
|
ct.ConnectStart("tcp", "127.0.0.1:3000")
|
|
|
|
ct.ConnectDone("tcp", "[::1]:3000", nil)
|
|
|
|
ct.ConnectDone("tcp", "127.0.0.1:3000", nil)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tts {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
tt.run()
|
2020-01-14 04:54:48 -08:00
|
|
|
spans := exp.spanMap["http.connect"]
|
2019-11-03 20:46:57 -03:00
|
|
|
|
|
|
|
if l := len(spans); l != 2 {
|
|
|
|
t.Fatalf("Expected 2 'http.connect' traces but found %d", l)
|
|
|
|
}
|
|
|
|
|
|
|
|
remotes := make(map[string]struct{})
|
|
|
|
for _, span := range spans {
|
|
|
|
if l := len(span.Attributes); l != 1 {
|
|
|
|
t.Fatalf("Expected 1 attribute on each span but found %d", l)
|
|
|
|
}
|
|
|
|
|
|
|
|
attr := span.Attributes[0]
|
|
|
|
if attr.Key != "http.remote" {
|
|
|
|
t.Fatalf("Expected attribute to be 'http.remote' but found %s", attr.Key)
|
|
|
|
}
|
|
|
|
remotes[attr.Value.Emit()] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
if l := len(remotes); l != 2 {
|
|
|
|
t.Fatalf("Expected 2 different 'http.remote' but found %d", l)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, remote := range []string{"127.0.0.1:3000", "[::1]:3000"} {
|
|
|
|
if _, ok := remotes[remote]; !ok {
|
|
|
|
t.Fatalf("Missing remote %s", remote)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-04-07 12:22:50 -04:00
|
|
|
|
|
|
|
func TestEndBeforeStartCreatesSpan(t *testing.T) {
|
|
|
|
exp := &testExporter{
|
|
|
|
spanMap: make(map[string][]*export.SpanData),
|
|
|
|
}
|
|
|
|
tp, _ := sdktrace.NewProvider(sdktrace.WithSyncer(exp), sdktrace.WithConfig(sdktrace.Config{DefaultSampler: sdktrace.AlwaysSample()}))
|
|
|
|
global.SetTraceProvider(tp)
|
|
|
|
|
|
|
|
tr := tp.Tracer("httptrace/client")
|
|
|
|
ctx, span := tr.Start(context.Background(), "test")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
ct := httptrace.NewClientTrace(ctx)
|
|
|
|
|
|
|
|
ct.DNSDone(nhtrace.DNSDoneInfo{})
|
|
|
|
ct.DNSStart(nhtrace.DNSStartInfo{Host: "example.com"})
|
|
|
|
|
|
|
|
getSpan := func(name string) *export.SpanData {
|
|
|
|
spans, ok := exp.spanMap[name]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("no spans found with the name %s, %v", name, exp.spanMap)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(spans) != 1 {
|
|
|
|
t.Fatalf("Expected exactly one span for %s but found %d", name, len(spans))
|
|
|
|
}
|
|
|
|
|
|
|
|
return spans[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
getSpan("http.dns")
|
|
|
|
|
|
|
|
}
|