123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- package clientinterceptors
- import (
- "context"
- "github.com/tal-tech/go-zero/core/opentelemetry"
- "go.opentelemetry.io/otel"
- "go.opentelemetry.io/otel/codes"
- "go.opentelemetry.io/otel/trace"
- "google.golang.org/grpc"
- grpc_codes "google.golang.org/grpc/codes"
- "google.golang.org/grpc/metadata"
- "google.golang.org/grpc/status"
- )
- func OpenTracingInterceptor() grpc.UnaryClientInterceptor {
- propagator := otel.GetTextMapPropagator()
- return func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
- if !opentelemetry.Enabled() {
- return invoker(ctx, method, req, reply, cc, opts...)
- }
- requestMetadata, _ := metadata.FromOutgoingContext(ctx)
- metadataCopy := requestMetadata.Copy()
- tr := otel.Tracer(opentelemetry.TraceName)
- name, attr := opentelemetry.SpanInfo(method, cc.Target())
- var span trace.Span
- ctx, span = tr.Start(ctx,
- name,
- trace.WithSpanKind(trace.SpanKindClient),
- trace.WithAttributes(attr...),
- )
- defer span.End()
- opentelemetry.Inject(ctx, propagator, &metadataCopy)
- ctx = metadata.NewOutgoingContext(ctx, metadataCopy)
- opentelemetry.MessageSent.Event(ctx, 1, req)
- err := invoker(ctx, method, req, reply, cc, opts...)
- opentelemetry.MessageReceived.Event(ctx, 1, reply)
- if err != nil {
- s, _ := status.FromError(err)
- span.SetStatus(codes.Error, s.Message())
- span.SetAttributes(opentelemetry.StatusCodeAttr(s.Code()))
- } else {
- span.SetAttributes(opentelemetry.StatusCodeAttr(grpc_codes.OK))
- }
- return err
- }
- }
- func StreamOpenTracingInterceptor() grpc.StreamClientInterceptor {
- propagator := otel.GetTextMapPropagator()
- return func(ctx context.Context, desc *grpc.StreamDesc, cc *grpc.ClientConn, method string, streamer grpc.Streamer, opts ...grpc.CallOption) (grpc.ClientStream, error) {
- if !opentelemetry.Enabled() {
- return streamer(ctx, desc, cc, method, opts...)
- }
- requestMetadata, _ := metadata.FromOutgoingContext(ctx)
- metadataCopy := requestMetadata.Copy()
- tr := otel.Tracer(opentelemetry.TraceName)
- name, attr := opentelemetry.SpanInfo(method, cc.Target())
- var span trace.Span
- ctx, span = tr.Start(
- ctx,
- name,
- trace.WithSpanKind(trace.SpanKindClient),
- trace.WithAttributes(attr...),
- )
- opentelemetry.Inject(ctx, propagator, &metadataCopy)
- ctx = metadata.NewOutgoingContext(ctx, metadataCopy)
- s, err := streamer(ctx, desc, cc, method, opts...)
- if err != nil {
- grpcStatus, _ := status.FromError(err)
- span.SetStatus(codes.Error, grpcStatus.Message())
- span.SetAttributes(opentelemetry.StatusCodeAttr(grpcStatus.Code()))
- span.End()
- return s, err
- }
- stream := opentelemetry.WrapClientStream(ctx, s, desc)
- go func() {
- err := <-stream.Finished
- if err != nil {
- s, _ := status.FromError(err)
- span.SetStatus(codes.Error, s.Message())
- span.SetAttributes(opentelemetry.StatusCodeAttr(s.Code()))
- } else {
- span.SetAttributes(opentelemetry.StatusCodeAttr(grpc_codes.OK))
- }
- span.End()
- }()
- return stream, nil
- }
- }
|