github.com/wangyougui/gf/v2@v2.6.5/net/gtrace/gtrace.go (about) 1 // Copyright GoFrame Author(https://goframe.org). All Rights Reserved. 2 // 3 // This Source Code Form is subject to the terms of the MIT License. 4 // If a copy of the MIT was not distributed with this file, 5 // You can obtain one at https://github.com/wangyougui/gf. 6 7 // Package gtrace provides convenience wrapping functionality for tracing feature using OpenTelemetry. 8 package gtrace 9 10 import ( 11 "context" 12 "os" 13 "strings" 14 15 "go.opentelemetry.io/otel" 16 "go.opentelemetry.io/otel/attribute" 17 "go.opentelemetry.io/otel/propagation" 18 semconv "go.opentelemetry.io/otel/semconv/v1.4.0" 19 "go.opentelemetry.io/otel/trace" 20 21 "github.com/wangyougui/gf/v2/container/gmap" 22 "github.com/wangyougui/gf/v2/container/gvar" 23 "github.com/wangyougui/gf/v2/errors/gcode" 24 "github.com/wangyougui/gf/v2/errors/gerror" 25 "github.com/wangyougui/gf/v2/internal/command" 26 "github.com/wangyougui/gf/v2/net/gipv4" 27 "github.com/wangyougui/gf/v2/net/gtrace/internal/provider" 28 "github.com/wangyougui/gf/v2/text/gstr" 29 "github.com/wangyougui/gf/v2/util/gconv" 30 ) 31 32 const ( 33 tracingCommonKeyIpIntranet = `ip.intranet` 34 tracingCommonKeyIpHostname = `hostname` 35 commandEnvKeyForMaxContentLogSize = "gf.gtrace.max.content.log.size" // To avoid too big tracing content. 36 commandEnvKeyForTracingInternal = "gf.gtrace.tracing.internal" // For detailed controlling for tracing content. 37 ) 38 39 var ( 40 intranetIps, _ = gipv4.GetIntranetIpArray() 41 intranetIpStr = strings.Join(intranetIps, ",") 42 hostname, _ = os.Hostname() 43 tracingInternal = true // tracingInternal enables tracing for internal type spans. 44 tracingMaxContentLogSize = 512 * 1024 // Max log size for request and response body, especially for HTTP/RPC request. 45 // defaultTextMapPropagator is the default propagator for context propagation between peers. 46 defaultTextMapPropagator = propagation.NewCompositeTextMapPropagator( 47 propagation.TraceContext{}, 48 propagation.Baggage{}, 49 ) 50 ) 51 52 func init() { 53 tracingInternal = gconv.Bool(command.GetOptWithEnv(commandEnvKeyForTracingInternal, "true")) 54 if maxContentLogSize := gconv.Int(command.GetOptWithEnv(commandEnvKeyForMaxContentLogSize)); maxContentLogSize > 0 { 55 tracingMaxContentLogSize = maxContentLogSize 56 } 57 // Default trace provider. 58 otel.SetTracerProvider(provider.New()) 59 CheckSetDefaultTextMapPropagator() 60 } 61 62 // IsUsingDefaultProvider checks and return if currently using default trace provider. 63 func IsUsingDefaultProvider() bool { 64 _, ok := otel.GetTracerProvider().(*provider.TracerProvider) 65 return ok 66 } 67 68 // IsTracingInternal returns whether tracing spans of internal components. 69 func IsTracingInternal() bool { 70 return tracingInternal 71 } 72 73 // MaxContentLogSize returns the max log size for request and response body, especially for HTTP/RPC request. 74 func MaxContentLogSize() int { 75 return tracingMaxContentLogSize 76 } 77 78 // CommonLabels returns common used attribute labels: 79 // ip.intranet, hostname. 80 func CommonLabels() []attribute.KeyValue { 81 return []attribute.KeyValue{ 82 attribute.String(tracingCommonKeyIpHostname, hostname), 83 attribute.String(tracingCommonKeyIpIntranet, intranetIpStr), 84 semconv.HostNameKey.String(hostname), 85 } 86 } 87 88 // CheckSetDefaultTextMapPropagator sets the default TextMapPropagator if it is not set previously. 89 func CheckSetDefaultTextMapPropagator() { 90 p := otel.GetTextMapPropagator() 91 if len(p.Fields()) == 0 { 92 otel.SetTextMapPropagator(GetDefaultTextMapPropagator()) 93 } 94 } 95 96 // GetDefaultTextMapPropagator returns the default propagator for context propagation between peers. 97 func GetDefaultTextMapPropagator() propagation.TextMapPropagator { 98 return defaultTextMapPropagator 99 } 100 101 // GetTraceID retrieves and returns TraceId from context. 102 // It returns an empty string is tracing feature is not activated. 103 func GetTraceID(ctx context.Context) string { 104 if ctx == nil { 105 return "" 106 } 107 traceID := trace.SpanContextFromContext(ctx).TraceID() 108 if traceID.IsValid() { 109 return traceID.String() 110 } 111 return "" 112 } 113 114 // GetSpanID retrieves and returns SpanId from context. 115 // It returns an empty string is tracing feature is not activated. 116 func GetSpanID(ctx context.Context) string { 117 if ctx == nil { 118 return "" 119 } 120 spanID := trace.SpanContextFromContext(ctx).SpanID() 121 if spanID.IsValid() { 122 return spanID.String() 123 } 124 return "" 125 } 126 127 // SetBaggageValue is a convenient function for adding one key-value pair to baggage. 128 // Note that it uses attribute.Any to set the key-value pair. 129 func SetBaggageValue(ctx context.Context, key string, value interface{}) context.Context { 130 return NewBaggage(ctx).SetValue(key, value) 131 } 132 133 // SetBaggageMap is a convenient function for adding map key-value pairs to baggage. 134 // Note that it uses attribute.Any to set the key-value pair. 135 func SetBaggageMap(ctx context.Context, data map[string]interface{}) context.Context { 136 return NewBaggage(ctx).SetMap(data) 137 } 138 139 // GetBaggageMap retrieves and returns the baggage values as map. 140 func GetBaggageMap(ctx context.Context) *gmap.StrAnyMap { 141 return NewBaggage(ctx).GetMap() 142 } 143 144 // GetBaggageVar retrieves value and returns a *gvar.Var for specified key from baggage. 145 func GetBaggageVar(ctx context.Context, key string) *gvar.Var { 146 return NewBaggage(ctx).GetVar(key) 147 } 148 149 // WithUUID injects custom trace id with UUID into context to propagate. 150 func WithUUID(ctx context.Context, uuid string) (context.Context, error) { 151 return WithTraceID(ctx, gstr.Replace(uuid, "-", "")) 152 } 153 154 // WithTraceID injects custom trace id into context to propagate. 155 func WithTraceID(ctx context.Context, traceID string) (context.Context, error) { 156 generatedTraceID, err := trace.TraceIDFromHex(traceID) 157 if err != nil { 158 return ctx, gerror.WrapCodef( 159 gcode.CodeInvalidParameter, 160 err, 161 `invalid custom traceID "%s", a traceID string should be composed with [0-f] and fixed length 32`, 162 traceID, 163 ) 164 } 165 sc := trace.SpanContextFromContext(ctx) 166 if !sc.HasTraceID() { 167 var span trace.Span 168 ctx, span = NewSpan(ctx, "gtrace.WithTraceID") 169 defer span.End() 170 sc = trace.SpanContextFromContext(ctx) 171 } 172 ctx = trace.ContextWithRemoteSpanContext(ctx, trace.NewSpanContext(trace.SpanContextConfig{ 173 TraceID: generatedTraceID, 174 SpanID: sc.SpanID(), 175 TraceFlags: sc.TraceFlags(), 176 TraceState: sc.TraceState(), 177 Remote: sc.IsRemote(), 178 })) 179 return ctx, nil 180 }