mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-24 03:47:19 +02:00
a228bafec5
* exporter(stackdriver): fix ExportSpans when ctx is not nil - problem: if ctx is not, calling cancel() will panic - nil Context already handled in newContextWithTimeout - add with test * update go.sum * mock traceserver for traceclient * make precommit * - respect option.Context when get google cred and traceclient * Revise with timeout, context - rename blackbox tests package to stackdriver_test - remove context is nil checking, add to initialization - add tests for timeout
160 lines
4.3 KiB
Go
160 lines
4.3 KiB
Go
// Copyright 2019, 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 stackdriver_test
|
|
|
|
import (
|
|
"context"
|
|
"flag"
|
|
"log"
|
|
"net"
|
|
"os"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
emptypb "github.com/golang/protobuf/ptypes/empty"
|
|
"github.com/stretchr/testify/assert"
|
|
"google.golang.org/api/option"
|
|
tracepb "google.golang.org/genproto/googleapis/devtools/cloudtrace/v2"
|
|
"google.golang.org/grpc"
|
|
|
|
"go.opentelemetry.io/otel/exporter/trace/stackdriver"
|
|
"go.opentelemetry.io/otel/global"
|
|
sdktrace "go.opentelemetry.io/otel/sdk/trace"
|
|
)
|
|
|
|
type mockTraceServer struct {
|
|
tracepb.TraceServiceServer
|
|
mu sync.Mutex
|
|
spansUploaded []*tracepb.Span
|
|
delay time.Duration
|
|
}
|
|
|
|
func (s *mockTraceServer) BatchWriteSpans(ctx context.Context, req *tracepb.BatchWriteSpansRequest) (*emptypb.Empty, error) {
|
|
var err error
|
|
s.mu.Lock()
|
|
select {
|
|
case <-ctx.Done():
|
|
err = ctx.Err()
|
|
case <-time.After(s.delay):
|
|
s.spansUploaded = append(s.spansUploaded, req.Spans...)
|
|
}
|
|
s.mu.Unlock()
|
|
return &emptypb.Empty{}, err
|
|
}
|
|
|
|
func (s *mockTraceServer) len() int {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
return len(s.spansUploaded)
|
|
}
|
|
|
|
// clientOpt is the option tests should use to connect to the test server.
|
|
// It is initialized by TestMain.
|
|
var clientOpt []option.ClientOption
|
|
|
|
var (
|
|
mockTrace mockTraceServer
|
|
)
|
|
|
|
func TestMain(m *testing.M) {
|
|
flag.Parse()
|
|
|
|
serv := grpc.NewServer()
|
|
tracepb.RegisterTraceServiceServer(serv, &mockTrace)
|
|
|
|
lis, err := net.Listen("tcp", "localhost:0")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
go func() {
|
|
_ = serv.Serve(lis)
|
|
}()
|
|
|
|
conn, err := grpc.Dial(lis.Addr().String(), grpc.WithInsecure())
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
clientOpt = []option.ClientOption{option.WithGRPCConn(conn)}
|
|
|
|
os.Exit(m.Run())
|
|
}
|
|
|
|
func TestExporter_ExportSpans(t *testing.T) {
|
|
// Initial test precondition
|
|
mockTrace.spansUploaded = nil
|
|
mockTrace.delay = 0
|
|
|
|
// Create StackDriver Exporter
|
|
exp, err := stackdriver.NewExporter(
|
|
stackdriver.WithProjectID("PROJECT_ID_NOT_REAL"),
|
|
stackdriver.WithTraceClientOptions(clientOpt),
|
|
)
|
|
assert.NoError(t, err)
|
|
|
|
tp, err := sdktrace.NewProvider(
|
|
sdktrace.WithConfig(sdktrace.Config{DefaultSampler: sdktrace.AlwaysSample()}),
|
|
sdktrace.WithBatcher(exp, // add following two options to ensure flush
|
|
sdktrace.WithScheduleDelayMillis(1),
|
|
sdktrace.WithMaxExportBatchSize(1),
|
|
))
|
|
assert.NoError(t, err)
|
|
|
|
global.SetTraceProvider(tp)
|
|
_, span := global.TraceProvider().GetTracer("test-tracer").Start(context.Background(), "test-span")
|
|
span.End()
|
|
assert.True(t, span.SpanContext().IsValid())
|
|
|
|
// wait exporter to flush
|
|
time.Sleep(20 * time.Millisecond)
|
|
assert.EqualValues(t, 1, mockTrace.len())
|
|
}
|
|
|
|
func TestExporter_Timeout(t *testing.T) {
|
|
// Initial test precondition
|
|
mockTrace.spansUploaded = nil
|
|
mockTrace.delay = 20 * time.Millisecond
|
|
var exportErrors []error
|
|
|
|
// Create StackDriver Exporter
|
|
exp, err := stackdriver.NewExporter(
|
|
stackdriver.WithProjectID("PROJECT_ID_NOT_REAL"),
|
|
stackdriver.WithTraceClientOptions(clientOpt),
|
|
stackdriver.WithTimeout(1*time.Millisecond),
|
|
stackdriver.WithOnError(func(err error) {
|
|
exportErrors = append(exportErrors, err)
|
|
}),
|
|
)
|
|
assert.NoError(t, err)
|
|
|
|
tp, err := sdktrace.NewProvider(
|
|
sdktrace.WithConfig(sdktrace.Config{DefaultSampler: sdktrace.AlwaysSample()}),
|
|
sdktrace.WithSyncer(exp))
|
|
assert.NoError(t, err)
|
|
|
|
global.SetTraceProvider(tp)
|
|
_, span := global.TraceProvider().GetTracer("test-tracer").Start(context.Background(), "test-span")
|
|
span.End()
|
|
assert.True(t, span.SpanContext().IsValid())
|
|
|
|
assert.EqualValues(t, 0, mockTrace.len())
|
|
if got, want := len(exportErrors), 1; got != want {
|
|
t.Fatalf("len(exportErrors) = %q; want %q", got, want)
|
|
}
|
|
if got, want := exportErrors[0].Error(), "rpc error: code = DeadlineExceeded desc = context deadline exceeded"; got != want {
|
|
t.Fatalf("err.Error() = %q; want %q", got, want)
|
|
}
|
|
}
|