mirror of
https://github.com/go-micro/go-micro.git
synced 2025-07-12 22:41:07 +02:00
* feat: more plugins * chore(ci): split out benchmarks Attempt to resolve too many open files in ci * chore(ci): split out benchmarks * fix(ci): Attempt to resolve too many open files in ci * fix: set DefaultX for cli flag and service option * fix: restore http broker * fix: default http broker * feat: full nats profile * chore: still ugly, not ready * fix: better initialization for profiles * fix(tests): comment out flaky listen tests * fix: disable benchmarks on gha * chore: cleanup, comments * chore: add nats config source
176 lines
5.2 KiB
Go
176 lines
5.2 KiB
Go
package opentelemetry
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"go-micro.dev/v5/client"
|
|
"go-micro.dev/v5/registry"
|
|
"go-micro.dev/v5/server"
|
|
"go.opentelemetry.io/otel/codes"
|
|
"go.opentelemetry.io/otel/trace"
|
|
)
|
|
|
|
// NewCallWrapper accepts an opentracing Tracer and returns a Call Wrapper.
|
|
func NewCallWrapper(opts ...Option) client.CallWrapper {
|
|
options := Options{}
|
|
for _, o := range opts {
|
|
o(&options)
|
|
}
|
|
return func(cf client.CallFunc) client.CallFunc {
|
|
return func(ctx context.Context, node *registry.Node, req client.Request, rsp interface{}, opts client.CallOptions) error {
|
|
if options.CallFilter != nil && options.CallFilter(ctx, req) {
|
|
return cf(ctx, node, req, rsp, opts)
|
|
}
|
|
name := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindClient),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, options.TraceProvider, name, spanOpts...)
|
|
defer span.End()
|
|
if err := cf(ctx, node, req, rsp, opts); err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// NewHandlerWrapper accepts an opentracing Tracer and returns a Handler Wrapper.
|
|
func NewHandlerWrapper(opts ...Option) server.HandlerWrapper {
|
|
options := Options{}
|
|
for _, o := range opts {
|
|
o(&options)
|
|
}
|
|
return func(h server.HandlerFunc) server.HandlerFunc {
|
|
return func(ctx context.Context, req server.Request, rsp interface{}) error {
|
|
if options.HandlerFilter != nil && options.HandlerFilter(ctx, req) {
|
|
return h(ctx, req, rsp)
|
|
}
|
|
name := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindServer),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, options.TraceProvider, name, spanOpts...)
|
|
defer span.End()
|
|
if err := h(ctx, req, rsp); err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// NewSubscriberWrapper accepts an opentracing Tracer and returns a Subscriber Wrapper.
|
|
func NewSubscriberWrapper(opts ...Option) server.SubscriberWrapper {
|
|
options := Options{}
|
|
for _, o := range opts {
|
|
o(&options)
|
|
}
|
|
return func(next server.SubscriberFunc) server.SubscriberFunc {
|
|
return func(ctx context.Context, msg server.Message) error {
|
|
if options.SubscriberFilter != nil && options.SubscriberFilter(ctx, msg) {
|
|
return next(ctx, msg)
|
|
}
|
|
name := "Sub from " + msg.Topic()
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindServer),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, options.TraceProvider, name, spanOpts...)
|
|
defer span.End()
|
|
if err := next(ctx, msg); err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// NewClientWrapper returns a client.Wrapper
|
|
// that adds monitoring to outgoing requests.
|
|
func NewClientWrapper(opts ...Option) client.Wrapper {
|
|
options := Options{}
|
|
for _, o := range opts {
|
|
o(&options)
|
|
}
|
|
return func(c client.Client) client.Client {
|
|
w := &clientWrapper{
|
|
Client: c,
|
|
tp: options.TraceProvider,
|
|
callFilter: options.CallFilter,
|
|
streamFilter: options.StreamFilter,
|
|
publishFilter: options.PublishFilter,
|
|
}
|
|
return w
|
|
}
|
|
}
|
|
|
|
type clientWrapper struct {
|
|
client.Client
|
|
|
|
tp trace.TracerProvider
|
|
callFilter CallFilter
|
|
streamFilter StreamFilter
|
|
publishFilter PublishFilter
|
|
}
|
|
|
|
func (w *clientWrapper) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
|
|
if w.callFilter != nil && w.callFilter(ctx, req) {
|
|
return w.Client.Call(ctx, req, rsp, opts...)
|
|
}
|
|
name := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindClient),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, w.tp, name, spanOpts...)
|
|
defer span.End()
|
|
if err := w.Client.Call(ctx, req, rsp, opts...); err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (w *clientWrapper) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
|
|
if w.streamFilter != nil && w.streamFilter(ctx, req) {
|
|
return w.Client.Stream(ctx, req, opts...)
|
|
}
|
|
name := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindClient),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, w.tp, name, spanOpts...)
|
|
defer span.End()
|
|
stream, err := w.Client.Stream(ctx, req, opts...)
|
|
if err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
}
|
|
return stream, err
|
|
}
|
|
|
|
func (w *clientWrapper) Publish(ctx context.Context, p client.Message, opts ...client.PublishOption) error {
|
|
if w.publishFilter != nil && w.publishFilter(ctx, p) {
|
|
return w.Client.Publish(ctx, p, opts...)
|
|
}
|
|
name := fmt.Sprintf("Pub to %s", p.Topic())
|
|
spanOpts := []trace.SpanStartOption{
|
|
trace.WithSpanKind(trace.SpanKindClient),
|
|
}
|
|
ctx, span := StartSpanFromContext(ctx, w.tp, name, spanOpts...)
|
|
defer span.End()
|
|
if err := w.Client.Publish(ctx, p, opts...); err != nil {
|
|
span.SetStatus(codes.Error, err.Error())
|
|
span.RecordError(err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|