pkg/rules/kratos/http/kratos_internal_setup.go (74 lines of code) (raw):

// Copyright (c) 2024 Alibaba Group Holding Ltd. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package http import ( "context" "os" _ "unsafe" "github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/api" kt "github.com/go-kratos/kratos/v2" "github.com/go-kratos/kratos/v2/middleware" "github.com/go-kratos/kratos/v2/transport" "github.com/go-kratos/kratos/v2/transport/grpc" "github.com/go-kratos/kratos/v2/transport/http" ) const OTEL_INSTRUMENTATION_KRATOS_EXPERIMENTAL_SPAN_ENABLE = "OTEL_INSTRUMENTATION_KRATOS_EXPERIMENTAL_SPAN_ENABLE" var kratosInternalInstrument = BuildKratosInternalInstrumenter() //go:linkname kratosNewHTTPServiceOnEnter github.com/go-kratos/kratos/v2/transport/http.kratosNewHTTPServiceOnEnter func kratosNewHTTPServiceOnEnter(call api.CallContext, opts ...http.ServerOption) { if os.Getenv(OTEL_INSTRUMENTATION_KRATOS_EXPERIMENTAL_SPAN_ENABLE) != "true" { return } opts = append(opts, AddHTTPMiddleware(ServerTracingMiddleWare())) call.SetParam(0, opts) } func AddHTTPMiddleware(m middleware.Middleware) http.ServerOption { return func(o *http.Server) { o.Use("*", m) } } func AddGRPCMiddleware(m middleware.Middleware) grpc.ServerOption { return func(o *grpc.Server) { o.Use("*", m) } } func ServerTracingMiddleWare() middleware.Middleware { return func(handler middleware.Handler) middleware.Handler { return func(ctx context.Context, req interface{}) (reply interface{}, err error) { if tr, ok := transport.FromServerContext(ctx); ok { serviceName, serviceId, serviceVersion := "", "", "" serviceEndpoint := make([]string, 0, 0) serviceMeta := make(map[string]string) app, hasApp := kt.FromContext(ctx) if hasApp { serviceName, serviceId, serviceVersion, serviceEndpoint = app.Name(), app.ID(), app.Version(), app.Endpoint() serviceMeta = app.Metadata() } var ( request kratosRequest sCtx context.Context ) request = kratosRequest{ serviceId: serviceId, serviceName: serviceName, serviceVersion: serviceVersion, serviceEndpoint: serviceEndpoint, serviceMeta: serviceMeta, } switch tr.Kind() { case transport.KindGRPC: request.protocolType = "grpc" sCtx = kratosInternalInstrument.Start(ctx, request) case transport.KindHTTP: request.protocolType = "http" sCtx = kratosInternalInstrument.Start(ctx, request) } defer func() { if err != nil { kratosInternalInstrument.End(sCtx, request, nil, err) } else { kratosInternalInstrument.End(sCtx, request, nil, err) } }() } return handler(ctx, req) } } }