pkg/inst-api-semconv/instrumenter/http/http_attrs_extractor.go (125 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"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/inst-api-semconv/instrumenter/net"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/inst-api/utils"
"go.opentelemetry.io/otel/attribute"
sdktrace "go.opentelemetry.io/otel/sdk/trace"
semconv "go.opentelemetry.io/otel/semconv/v1.30.0"
"go.opentelemetry.io/otel/trace"
"strings"
)
// TODO: remove server.address and put it into NetworkAttributesExtractor
type HttpCommonAttrsExtractor[REQUEST any, RESPONSE any, GETTER1 HttpCommonAttrsGetter[REQUEST, RESPONSE], GETTER2 net.NetworkAttrsGetter[REQUEST, RESPONSE]] struct {
HttpGetter GETTER1
NetGetter GETTER2
AttributesFilter func(attrs []attribute.KeyValue) []attribute.KeyValue
}
func (h *HttpCommonAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]) OnStart(attributes []attribute.KeyValue, parentContext context.Context, request REQUEST) ([]attribute.KeyValue, context.Context) {
attributes = append(attributes, attribute.KeyValue{
Key: semconv.HTTPRequestMethodKey,
Value: attribute.StringValue(h.HttpGetter.GetRequestMethod(request)),
})
return attributes, parentContext
}
func (h *HttpCommonAttrsExtractor[REQUEST, RESPONSE, GETTER, GETTER2]) OnEnd(attributes []attribute.KeyValue, context context.Context, request REQUEST, response RESPONSE, err error) ([]attribute.KeyValue, context.Context) {
statusCode := h.HttpGetter.GetHttpResponseStatusCode(request, response, err)
protocolName := h.NetGetter.GetNetworkProtocolName(request, response)
protocolVersion := h.NetGetter.GetNetworkProtocolVersion(request, response)
attributes = append(attributes, attribute.KeyValue{
Key: semconv.HTTPResponseStatusCodeKey,
Value: attribute.IntValue(statusCode),
}, attribute.KeyValue{
Key: semconv.NetworkProtocolNameKey,
Value: attribute.StringValue(protocolName),
}, attribute.KeyValue{
Key: semconv.NetworkProtocolVersionKey,
Value: attribute.StringValue(protocolVersion),
})
errorType := h.HttpGetter.GetErrorType(request, response, err)
if errorType != "" {
attributes = append(attributes, attribute.KeyValue{Key: semconv.ErrorTypeKey, Value: attribute.StringValue(errorType)})
}
return attributes, context
}
type HttpClientAttrsExtractor[REQUEST any, RESPONSE any, GETTER1 HttpClientAttrsGetter[REQUEST, RESPONSE], GETTER2 net.NetworkAttrsGetter[REQUEST, RESPONSE]] struct {
Base HttpCommonAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]
NetworkExtractor net.NetworkAttrsExtractor[REQUEST, RESPONSE, GETTER2]
}
func (h *HttpClientAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]) OnStart(attributes []attribute.KeyValue, parentContext context.Context, request REQUEST) ([]attribute.KeyValue, context.Context) {
attributes, parentContext = h.Base.OnStart(attributes, parentContext, request)
attributes, parentContext = h.NetworkExtractor.OnStart(attributes, parentContext, request)
fullUrl := h.Base.HttpGetter.GetUrlFull(request)
// TODO: add resend count
attributes = append(attributes, attribute.KeyValue{
Key: semconv.URLFullKey,
Value: attribute.StringValue(fullUrl),
}, attribute.KeyValue{
Key: semconv.ServerAddressKey,
Value: attribute.StringValue(h.Base.HttpGetter.GetServerAddress(request)),
}, attribute.KeyValue{
Key: semconv.ServerPortKey,
Value: attribute.IntValue(h.Base.HttpGetter.GetServerPort(request)),
})
if h.Base.AttributesFilter != nil {
attributes = h.Base.AttributesFilter(attributes)
}
return attributes, parentContext
}
func (h *HttpClientAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]) OnEnd(attributes []attribute.KeyValue, context context.Context, request REQUEST, response RESPONSE, err error) ([]attribute.KeyValue, context.Context) {
attributes, context = h.Base.OnEnd(attributes, context, request, response, err)
attributes, context = h.NetworkExtractor.OnEnd(attributes, context, request, response, err)
if h.Base.AttributesFilter != nil {
attributes = h.Base.AttributesFilter(attributes)
}
return attributes, context
}
func (h *HttpClientAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]) GetSpanKey() attribute.Key {
return utils.HTTP_CLIENT_KEY
}
type HttpServerAttrsExtractor[REQUEST any, RESPONSE any, GETTER1 HttpServerAttrsGetter[REQUEST, RESPONSE], GETTER2 net.NetworkAttrsGetter[REQUEST, RESPONSE], GETTER3 net.UrlAttrsGetter[REQUEST]] struct {
Base HttpCommonAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2]
NetworkExtractor net.NetworkAttrsExtractor[REQUEST, RESPONSE, GETTER2]
UrlExtractor net.UrlAttrsExtractor[REQUEST, RESPONSE, GETTER3]
}
func (h *HttpServerAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2, GETTER3]) OnStart(attributes []attribute.KeyValue, parentContext context.Context, request REQUEST) ([]attribute.KeyValue, context.Context) {
attributes, parentContext = h.Base.OnStart(attributes, parentContext, request)
attributes, parentContext = h.UrlExtractor.OnStart(attributes, parentContext, request)
userAgent := h.Base.HttpGetter.GetHttpRequestHeader(request, "User-Agent")
var firstUserAgent string
if len(userAgent) > 0 {
firstUserAgent = userAgent[0]
} else {
firstUserAgent = ""
}
attributes = append(attributes, attribute.KeyValue{
Key: semconv.UserAgentOriginalKey,
Value: attribute.StringValue(firstUserAgent),
})
if h.Base.AttributesFilter != nil {
attributes = h.Base.AttributesFilter(attributes)
}
return attributes, parentContext
}
func (h *HttpServerAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2, GETTER3]) OnEnd(attributes []attribute.KeyValue, context context.Context, request REQUEST, response RESPONSE, err error) ([]attribute.KeyValue, context.Context) {
attributes, context = h.Base.OnEnd(attributes, context, request, response, err)
attributes, context = h.UrlExtractor.OnEnd(attributes, context, request, response, err)
attributes, context = h.NetworkExtractor.OnEnd(attributes, context, request, response, err)
span := trace.SpanFromContext(context)
localRootSpan, ok := span.(sdktrace.ReadOnlySpan)
if ok && span.IsRecording() {
route := h.Base.HttpGetter.GetHttpRoute(request)
if !strings.Contains(localRootSpan.Name(), route) {
route = localRootSpan.Name()
}
attributes = append(attributes, attribute.KeyValue{
Key: semconv.HTTPRouteKey,
Value: attribute.StringValue(route),
})
}
if h.Base.AttributesFilter != nil {
attributes = h.Base.AttributesFilter(attributes)
}
return attributes, context
}
func (h *HttpServerAttrsExtractor[REQUEST, RESPONSE, GETTER1, GETTER2, GETTER3]) GetSpanKey() attribute.Key {
return utils.HTTP_SERVER_KEY
}