2020-01-29 18:45:11 +03:00
|
|
|
package memory
|
2020-01-18 13:20:46 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2020-01-30 14:39:00 +03:00
|
|
|
"github.com/micro/go-micro/v2/debug/trace"
|
|
|
|
"github.com/micro/go-micro/v2/util/ring"
|
2020-01-18 13:20:46 +03:00
|
|
|
)
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
type Tracer struct {
|
|
|
|
opts trace.Options
|
2020-01-18 13:20:46 +03:00
|
|
|
|
|
|
|
// ring buffer of traces
|
|
|
|
buffer *ring.Buffer
|
|
|
|
}
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
func (t *Tracer) Read(opts ...trace.ReadOption) ([]*trace.Span, error) {
|
|
|
|
var options trace.ReadOptions
|
2020-01-25 00:24:51 +03:00
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
sp := t.buffer.Get(t.buffer.Size())
|
|
|
|
|
2020-04-22 16:10:59 +03:00
|
|
|
spans := make([]*trace.Span, 0, len(sp))
|
2020-01-25 00:24:51 +03:00
|
|
|
|
|
|
|
for _, span := range sp {
|
2020-01-29 18:45:11 +03:00
|
|
|
val := span.Value.(*trace.Span)
|
2020-01-25 00:24:51 +03:00
|
|
|
// skip if trace id is specified and doesn't match
|
|
|
|
if len(options.Trace) > 0 && val.Trace != options.Trace {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
spans = append(spans, val)
|
|
|
|
}
|
|
|
|
|
|
|
|
return spans, nil
|
2020-01-18 13:20:46 +03:00
|
|
|
}
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
func (t *Tracer) Start(ctx context.Context, name string) (context.Context, *trace.Span) {
|
|
|
|
span := &trace.Span{
|
2020-01-18 13:20:46 +03:00
|
|
|
Name: name,
|
|
|
|
Trace: uuid.New().String(),
|
|
|
|
Id: uuid.New().String(),
|
|
|
|
Started: time.Now(),
|
|
|
|
Metadata: make(map[string]string),
|
|
|
|
}
|
|
|
|
|
|
|
|
// return span if no context
|
|
|
|
if ctx == nil {
|
2020-02-06 20:22:16 +03:00
|
|
|
return trace.ToContext(context.Background(), span.Trace, span.Id), span
|
2020-01-18 13:20:46 +03:00
|
|
|
}
|
2020-02-06 20:22:16 +03:00
|
|
|
traceID, parentSpanID, ok := trace.FromContext(ctx)
|
|
|
|
// If the trace can not be found in the header,
|
|
|
|
// that means this is where the trace is created.
|
2020-01-18 13:20:46 +03:00
|
|
|
if !ok {
|
2020-02-06 20:22:16 +03:00
|
|
|
return trace.ToContext(ctx, span.Trace, span.Id), span
|
2020-01-18 13:20:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// set trace id
|
2020-02-06 20:22:16 +03:00
|
|
|
span.Trace = traceID
|
2020-01-18 13:20:46 +03:00
|
|
|
// set parent
|
2020-02-06 20:22:16 +03:00
|
|
|
span.Parent = parentSpanID
|
2020-01-18 13:20:46 +03:00
|
|
|
|
2020-02-06 20:22:16 +03:00
|
|
|
// return the span
|
|
|
|
return trace.ToContext(ctx, span.Trace, span.Id), span
|
2020-01-18 13:20:46 +03:00
|
|
|
}
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
func (t *Tracer) Finish(s *trace.Span) error {
|
2020-01-18 13:20:46 +03:00
|
|
|
// set finished time
|
2020-01-25 00:24:51 +03:00
|
|
|
s.Duration = time.Since(s.Started)
|
2020-01-18 13:20:46 +03:00
|
|
|
// save the span
|
|
|
|
t.buffer.Put(s)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
func NewTracer(opts ...trace.Option) trace.Tracer {
|
|
|
|
var options trace.Options
|
2020-01-18 13:20:46 +03:00
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
2020-01-29 18:45:11 +03:00
|
|
|
return &Tracer{
|
2020-01-18 13:20:46 +03:00
|
|
|
opts: options,
|
2020-02-06 20:22:16 +03:00
|
|
|
// the last 256 requests
|
|
|
|
buffer: ring.New(256),
|
2020-01-18 13:20:46 +03:00
|
|
|
}
|
|
|
|
}
|