192 lines
4.9 KiB
Go
192 lines
4.9 KiB
Go
|
package instana
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
|
||
|
sensor "github.com/instana/go-sensor"
|
||
|
"github.com/opentracing/opentracing-go"
|
||
|
"github.com/opentracing/opentracing-go/log"
|
||
|
"go.unistack.org/micro/v4/metadata"
|
||
|
"go.unistack.org/micro/v4/options"
|
||
|
"go.unistack.org/micro/v4/tracer"
|
||
|
)
|
||
|
|
||
|
var _ tracer.Tracer = &Tracer{}
|
||
|
|
||
|
type Tracer struct {
|
||
|
opts tracer.Options
|
||
|
sensor sensor.Tracer
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) Name() string {
|
||
|
return ot.opts.Name
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) Flush(ctx context.Context) error {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) Init(opts ...options.Option) error {
|
||
|
for _, o := range opts {
|
||
|
o(&ot.opts)
|
||
|
}
|
||
|
|
||
|
sensorOptions := sensor.DefaultOptions()
|
||
|
if v, ok := ot.opts.Context.Value(tracerOptionsKey{}).(*sensor.Options); ok && v != nil {
|
||
|
sensorOptions = v
|
||
|
}
|
||
|
|
||
|
ot.sensor = sensor.NewSensorWithTracer(sensor.NewTracerWithOptions(sensorOptions))
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) Start(ctx context.Context, name string, opts ...options.Option) (context.Context, tracer.Span) {
|
||
|
options := tracer.NewSpanOptions(opts...)
|
||
|
var span opentracing.Span
|
||
|
switch options.Kind {
|
||
|
case tracer.SpanKindInternal, tracer.SpanKindUnspecified:
|
||
|
ctx, span = ot.startSpanFromContext(ctx, name)
|
||
|
case tracer.SpanKindClient, tracer.SpanKindProducer:
|
||
|
ctx, span = ot.startSpanFromOutgoingContext(ctx, name)
|
||
|
case tracer.SpanKindServer, tracer.SpanKindConsumer:
|
||
|
ctx, span = ot.startSpanFromIncomingContext(ctx, name)
|
||
|
}
|
||
|
return ctx, &otSpan{span: span, opts: options}
|
||
|
}
|
||
|
|
||
|
type otSpan struct {
|
||
|
span opentracing.Span
|
||
|
sensor sensor.Tracer
|
||
|
topts tracer.Options
|
||
|
opts tracer.SpanOptions
|
||
|
status tracer.SpanStatus
|
||
|
statusMsg string
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) SetStatus(st tracer.SpanStatus, msg string) {
|
||
|
switch st {
|
||
|
case tracer.SpanStatusError:
|
||
|
os.span.SetTag("error", true)
|
||
|
}
|
||
|
os.status = st
|
||
|
os.statusMsg = msg
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) Status() (tracer.SpanStatus, string) {
|
||
|
return os.status, os.statusMsg
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) Tracer() tracer.Tracer {
|
||
|
return &Tracer{sensor: os.sensor, opts: os.topts}
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) Finish(opts ...options.Option) {
|
||
|
if len(os.opts.Labels) > 0 {
|
||
|
os.span.LogKV(os.opts.Labels...)
|
||
|
}
|
||
|
os.span.Finish()
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) AddEvent(name string, opts ...options.Option) {
|
||
|
os.span.LogFields(log.Event(name))
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) Context() context.Context {
|
||
|
return opentracing.ContextWithSpan(context.Background(), os.span)
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) SetName(name string) {
|
||
|
os.span = os.span.SetOperationName(name)
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) SetLabels(labels ...interface{}) {
|
||
|
os.opts.Labels = labels
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) Kind() tracer.SpanKind {
|
||
|
return os.opts.Kind
|
||
|
}
|
||
|
|
||
|
func (os *otSpan) AddLabels(labels ...interface{}) {
|
||
|
os.opts.Labels = append(os.opts.Labels, labels...)
|
||
|
}
|
||
|
|
||
|
func NewTracer(opts ...options.Option) *Tracer {
|
||
|
options := tracer.NewOptions(opts...)
|
||
|
return &Tracer{opts: options}
|
||
|
}
|
||
|
|
||
|
func spanFromContext(ctx context.Context) (opentracing.Span, bool) {
|
||
|
return sensor.SpanFromContext(ctx)
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) startSpanFromContext(ctx context.Context, name string, opts ...opentracing.StartSpanOption) (context.Context, opentracing.Span) {
|
||
|
if parentSpan := opentracing.SpanFromContext(ctx); parentSpan != nil {
|
||
|
opts = append(opts, opentracing.ChildOf(parentSpan.Context()))
|
||
|
}
|
||
|
|
||
|
md := metadata.New(1)
|
||
|
|
||
|
sp := ot.sensor.StartSpan(name, opts...)
|
||
|
if err := sp.Tracer().Inject(sp.Context(), opentracing.TextMap, opentracing.TextMapCarrier(md)); err != nil {
|
||
|
return nil, nil
|
||
|
}
|
||
|
|
||
|
ctx = opentracing.ContextWithSpan(ctx, sp)
|
||
|
|
||
|
return ctx, sp
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) startSpanFromOutgoingContext(ctx context.Context, name string, opts ...opentracing.StartSpanOption) (context.Context, opentracing.Span) {
|
||
|
var parentCtx opentracing.SpanContext
|
||
|
|
||
|
md, ok := metadata.FromOutgoingContext(ctx)
|
||
|
if ok && md != nil {
|
||
|
if spanCtx, err := ot.sensor.Extract(opentracing.TextMap, opentracing.TextMapCarrier(md)); err == nil && ok {
|
||
|
parentCtx = spanCtx
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if parentCtx != nil {
|
||
|
opts = append(opts, opentracing.ChildOf(parentCtx))
|
||
|
}
|
||
|
|
||
|
nmd := metadata.Copy(md)
|
||
|
|
||
|
sp := ot.sensor.StartSpan(name, opts...)
|
||
|
if err := sp.Tracer().Inject(sp.Context(), opentracing.TextMap, opentracing.TextMapCarrier(nmd)); err != nil {
|
||
|
return nil, nil
|
||
|
}
|
||
|
|
||
|
ctx = metadata.NewOutgoingContext(opentracing.ContextWithSpan(ctx, sp), nmd)
|
||
|
|
||
|
return ctx, sp
|
||
|
}
|
||
|
|
||
|
func (ot *Tracer) startSpanFromIncomingContext(ctx context.Context, name string, opts ...opentracing.StartSpanOption) (context.Context, opentracing.Span) {
|
||
|
var parentCtx opentracing.SpanContext
|
||
|
|
||
|
md, ok := metadata.FromIncomingContext(ctx)
|
||
|
if ok && md != nil {
|
||
|
if spanCtx, err := ot.sensor.Extract(opentracing.TextMap, opentracing.TextMapCarrier(md)); err == nil {
|
||
|
parentCtx = spanCtx
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if parentCtx != nil {
|
||
|
opts = append(opts, opentracing.ChildOf(parentCtx))
|
||
|
}
|
||
|
|
||
|
nmd := metadata.Copy(md)
|
||
|
|
||
|
sp := ot.sensor.StartSpan(name, opts...)
|
||
|
if err := sp.Tracer().Inject(sp.Context(), opentracing.TextMap, opentracing.TextMapCarrier(nmd)); err != nil {
|
||
|
return nil, nil
|
||
|
}
|
||
|
|
||
|
ctx = metadata.NewIncomingContext(opentracing.ContextWithSpan(ctx, sp), nmd)
|
||
|
|
||
|
return ctx, sp
|
||
|
}
|