|
|
|
package detect
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/moby/buildkit/util/bklog"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"go.opentelemetry.io/otel/sdk/resource"
|
|
|
|
sdktrace "go.opentelemetry.io/otel/sdk/trace"
|
|
|
|
semconv "go.opentelemetry.io/otel/semconv/v1.12.0"
|
|
|
|
"go.opentelemetry.io/otel/trace"
|
|
|
|
)
|
|
|
|
|
|
|
|
type ExporterDetector func() (sdktrace.SpanExporter, error)
|
|
|
|
|
|
|
|
type detector struct {
|
|
|
|
f ExporterDetector
|
|
|
|
priority int
|
|
|
|
}
|
|
|
|
|
|
|
|
var ServiceName string
|
|
|
|
var Recorder *TraceRecorder
|
|
|
|
|
|
|
|
var detectors map[string]detector
|
|
|
|
var once sync.Once
|
|
|
|
var tp trace.TracerProvider
|
|
|
|
var exporter sdktrace.SpanExporter
|
|
|
|
var closers []func(context.Context) error
|
|
|
|
var err error
|
|
|
|
|
|
|
|
func Register(name string, exp ExporterDetector, priority int) {
|
|
|
|
if detectors == nil {
|
|
|
|
detectors = map[string]detector{}
|
|
|
|
}
|
|
|
|
detectors[name] = detector{
|
|
|
|
f: exp,
|
|
|
|
priority: priority,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func detectExporter() (sdktrace.SpanExporter, error) {
|
|
|
|
if n := os.Getenv("OTEL_TRACES_EXPORTER"); n != "" {
|
|
|
|
d, ok := detectors[n]
|
|
|
|
if !ok {
|
|
|
|
if n == "none" {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, errors.Errorf("unsupported opentelemetry tracer %v", n)
|
|
|
|
}
|
|
|
|
return d.f()
|
|
|
|
}
|
|
|
|
arr := make([]detector, 0, len(detectors))
|
|
|
|
for _, d := range detectors {
|
|
|
|
arr = append(arr, d)
|
|
|
|
}
|
|
|
|
sort.Slice(arr, func(i, j int) bool {
|
|
|
|
return arr[i].priority < arr[j].priority
|
|
|
|
})
|
|
|
|
for _, d := range arr {
|
|
|
|
exp, err := d.f()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if exp != nil {
|
|
|
|
return exp, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getExporter() (sdktrace.SpanExporter, error) {
|
|
|
|
exp, err := detectExporter()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if exp != nil {
|
|
|
|
exp = &threadSafeExporterWrapper{
|
|
|
|
exporter: exp,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if Recorder != nil {
|
|
|
|
Recorder.SpanExporter = exp
|
|
|
|
exp = Recorder
|
|
|
|
}
|
|
|
|
return exp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func detect() error {
|
|
|
|
tp = trace.NewNoopTracerProvider()
|
|
|
|
|
|
|
|
exp, err := getExporter()
|
|
|
|
if err != nil || exp == nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// enable log with traceID when valid exporter
|
|
|
|
bklog.EnableLogWithTraceID(true)
|
|
|
|
|
|
|
|
res, err := resource.Detect(context.Background(), serviceNameDetector{})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
res, err = resource.Merge(resource.Default(), res)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
sp := sdktrace.NewBatchSpanProcessor(exp)
|
|
|
|
|
|
|
|
if Recorder != nil {
|
|
|
|
Recorder.flush = sp.ForceFlush
|
|
|
|
}
|
|
|
|
|
|
|
|
sdktp := sdktrace.NewTracerProvider(sdktrace.WithSpanProcessor(sp), sdktrace.WithResource(res))
|
|
|
|
closers = append(closers, sdktp.Shutdown)
|
|
|
|
|
|
|
|
exporter = exp
|
|
|
|
tp = sdktp
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TracerProvider() (trace.TracerProvider, error) {
|
|
|
|
once.Do(func() {
|
|
|
|
if err1 := detect(); err1 != nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
})
|
|
|
|
b, _ := strconv.ParseBool(os.Getenv("OTEL_IGNORE_ERROR"))
|
|
|
|
if err != nil && !b {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return tp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Exporter() (sdktrace.SpanExporter, error) {
|
|
|
|
_, err := TracerProvider()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return exporter, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Shutdown(ctx context.Context) error {
|
|
|
|
for _, c := range closers {
|
|
|
|
if err := c(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type serviceNameDetector struct{}
|
|
|
|
|
|
|
|
func (serviceNameDetector) Detect(ctx context.Context) (*resource.Resource, error) {
|
|
|
|
return resource.StringDetector(
|
|
|
|
semconv.SchemaURL,
|
|
|
|
semconv.ServiceNameKey,
|
|
|
|
func() (string, error) {
|
|
|
|
if n := os.Getenv("OTEL_SERVICE_NAME"); n != "" {
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
if ServiceName != "" {
|
|
|
|
return ServiceName, nil
|
|
|
|
}
|
|
|
|
return filepath.Base(os.Args[0]), nil
|
|
|
|
},
|
|
|
|
).Detect(ctx)
|
|
|
|
}
|