From d641c90718681485a330d862c0b2cc7beccbfe87 Mon Sep 17 00:00:00 2001 From: Kevin Boonstra <23527353+Strazz1337@users.noreply.github.com> Date: Sat, 12 Oct 2024 14:21:22 +0200 Subject: [PATCH 1/7] fix(spans): adhere attribute name to otel semver --- compression_handler.go | 7 ++++--- nethttp_request_adapter.go | 23 ++++++++++++----------- redirect_handler.go | 3 ++- retry_handler.go | 4 ++-- 4 files changed, 20 insertions(+), 17 deletions(-) diff --git a/compression_handler.go b/compression_handler.go index 61d7ba7..fec1eb5 100644 --- a/compression_handler.go +++ b/compression_handler.go @@ -10,6 +10,7 @@ import ( abstractions "github.com/microsoft/kiota-abstractions-go" "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" + semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -104,7 +105,7 @@ func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, r req.ContentLength = int64(size) if span != nil { - span.SetAttributes(attribute.Int64("http.request_content_length", req.ContentLength)) + span.SetAttributes(semconv.HTTPRequestBodySize(int(req.ContentLength))) } // Sending request with compressed body @@ -120,8 +121,8 @@ func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, r req.ContentLength = unCompressedContentLength if span != nil { - span.SetAttributes(attribute.Int64("http.request_content_length", req.ContentLength), - attribute.Int("http.request_content_length", 415)) + span.SetAttributes(semconv.HTTPRequestBodySize(int(req.ContentLength)), + semconv.HTTPResponseStatusCode(415)) } return pipeline.Next(req, middlewareIndex) diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index e668c4f..1eb345d 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -18,6 +18,7 @@ import ( "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" "go.opentelemetry.io/otel/codes" + semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -148,15 +149,15 @@ func (a *NetHttpRequestAdapter) getHttpResponseMessage(ctx context.Context, requ contentLenHeader := response.Header.Get("Content-Length") if contentLenHeader != "" { contentLen, _ := strconv.Atoi(contentLenHeader) - spanForAttributes.SetAttributes(attribute.Int("http.response_content_length", contentLen)) + spanForAttributes.SetAttributes(semconv.HTTPResponseBodySize(contentLen)) } contentTypeHeader := response.Header.Get("Content-Type") if contentTypeHeader != "" { - spanForAttributes.SetAttributes(attribute.String("http.response_content_type", contentTypeHeader)) + spanForAttributes.SetAttributes(attribute.String("http.response.header.content-type", contentTypeHeader)) } spanForAttributes.SetAttributes( - attribute.Int("http.status_code", response.StatusCode), - attribute.String("http.flavor", response.Proto), + semconv.HTTPResponseStatusCode(response.StatusCode), + semconv.NetworkProtocolName(response.Proto), ) } return a.retryCAEResponseIfRequired(ctx, response, requestInfo, claims, spanForAttributes) @@ -253,19 +254,19 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con if spanForAttributes == nil { spanForAttributes = span } - spanForAttributes.SetAttributes(attribute.String("http.method", requestInfo.Method.String())) + spanForAttributes.SetAttributes(semconv.HTTPRequestMethodKey.String(requestInfo.Method.String())) uri, err := requestInfo.GetUri() if err != nil { spanForAttributes.RecordError(err) return nil, err } spanForAttributes.SetAttributes( - attribute.String("http.scheme", uri.Scheme), - attribute.String("http.host", uri.Host), + semconv.ServerAddress(uri.Scheme), + semconv.URLScheme(uri.Host), ) if a.observabilityOptions.IncludeEUIIAttributes { - spanForAttributes.SetAttributes(attribute.String("http.uri", uri.String())) + spanForAttributes.SetAttributes(semconv.URLFull(uri.String())) } request, err := nethttp.NewRequestWithContext(ctx, requestInfo.Method.String(), uri.String(), nil) @@ -290,14 +291,14 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con } if request.Header.Get("Content-Type") != "" { spanForAttributes.SetAttributes( - attribute.String("http.request_content_type", request.Header.Get("Content-Type")), + attribute.String("http.request.header.content-type", request.Header.Get("Content-Type")), ) } if request.Header.Get("Content-Length") != "" { contentLenVal, _ := strconv.Atoi(request.Header.Get("Content-Length")) request.ContentLength = int64(contentLenVal) spanForAttributes.SetAttributes( - attribute.Int("http.request_content_length", contentLenVal), + semconv.HTTPRequestBodySize(contentLenVal), ) } } @@ -313,7 +314,7 @@ func (a *NetHttpRequestAdapter) startTracingSpan(ctx context.Context, requestInf decodedUriTemplate := decodeUriEncodedString(requestInfo.UrlTemplate, []byte{'-', '.', '~', '$'}) telemetryPathValue := queryParametersCleanupRegex.ReplaceAll([]byte(decodedUriTemplate), []byte("")) ctx, span := otel.GetTracerProvider().Tracer(a.observabilityOptions.GetTracerInstrumentationName()).Start(ctx, methodName+" - "+string(telemetryPathValue)) - span.SetAttributes(attribute.String("http.uri_template", decodedUriTemplate)) + span.SetAttributes(attribute.String("url.uri_template", decodedUriTemplate)) return ctx, span } diff --git a/redirect_handler.go b/redirect_handler.go index a9c2def..eb71059 100644 --- a/redirect_handler.go +++ b/redirect_handler.go @@ -11,6 +11,7 @@ import ( abs "github.com/microsoft/kiota-abstractions-go" "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" + semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -120,7 +121,7 @@ func (middleware RedirectHandler) redirectRequest(ctx context.Context, pipeline if observabilityName != "" { ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RedirectHandler_Intercept - redirect "+fmt.Sprint(redirectCount)) span.SetAttributes(attribute.Int("com.microsoft.kiota.handler.redirect.count", redirectCount), - attribute.Int("http.status_code", response.StatusCode), + semconv.HTTPResponseStatusCode(response.StatusCode), ) defer span.End() redirectRequest = redirectRequest.WithContext(ctx) diff --git a/retry_handler.go b/retry_handler.go index f366396..2b65f36 100644 --- a/retry_handler.go +++ b/retry_handler.go @@ -12,6 +12,7 @@ import ( abs "github.com/microsoft/kiota-abstractions-go" "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" + semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -144,9 +145,8 @@ func (middleware RetryHandler) retryRequest(ctx context.Context, pipeline Pipeli ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept - attempt "+fmt.Sprint(executionCount)) span.SetAttributes(attribute.Int("http.request.resend_count", executionCount), - attribute.Int("http.status_code", resp.StatusCode), + semconv.HTTPResponseStatusCode(resp.StatusCode), attribute.Float64("http.request.resend_delay", delay.Seconds()), - ) defer span.End() req = req.WithContext(ctx) From 7f78df40bff597cfe5a3f0b30a7d830b7b01e6c4 Mon Sep 17 00:00:00 2001 From: Kevin Boonstra <23527353+Strazz1337@users.noreply.github.com> Date: Sat, 12 Oct 2024 14:25:19 +0200 Subject: [PATCH 2/7] chore: bump version to v1.4.6 --- CHANGELOG.md | 6 ++++++ nethttp_request_adapter.go | 2 +- user_agent_handler.go | 2 +- 3 files changed, 8 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 165ff1f..0b8abad 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +## [1.4.6] - 2024-10-12 + +### Changed + +- Updated HTTP span attributes to comply with updated OpenTelemetry semantic conventions. [#182](https://github.com/microsoft/kiota-http-go/issues/182) + ## [1.4.5] - 2024-09-03 ### Changed diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index 1eb345d..23a025f 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -178,7 +178,7 @@ func (a *NetHttpRequestAdapter) retryCAEResponseIfRequired(ctx context.Context, authenticateHeaderVal := response.Header.Get("WWW-Authenticate") if authenticateHeaderVal != "" && reBearer.Match([]byte(authenticateHeaderVal)) { span.AddEvent(AuthenticateChallengedEventKey) - spanForAttributes.SetAttributes(attribute.Int("http.retry_count", 1)) + spanForAttributes.SetAttributes(semconv.HTTPRequestResendCount(1)) responseClaims := "" parametersRaw := string(reBearer.ReplaceAll([]byte(authenticateHeaderVal), []byte(""))) parameters := strings.Split(parametersRaw, ",") diff --git a/user_agent_handler.go b/user_agent_handler.go index 3f94bf2..51af61b 100644 --- a/user_agent_handler.go +++ b/user_agent_handler.go @@ -42,7 +42,7 @@ func NewUserAgentHandlerOptions() *UserAgentHandlerOptions { return &UserAgentHandlerOptions{ Enabled: true, ProductName: "kiota-go", - ProductVersion: "1.4.5", + ProductVersion: "1.4.6", } } From f94251c0dea6470906c076deb1fed861a57dde5b Mon Sep 17 00:00:00 2001 From: Kevin Boonstra <23527353+Strazz1337@users.noreply.github.com> Date: Fri, 13 Dec 2024 15:20:00 +0100 Subject: [PATCH 3/7] refactor: use separate file for semantic conventions --- CHANGELOG.md | 7 +- compression_handler.go | 323 +++++++++++++++--------------- nethttp_request_adapter.go | 23 ++- redirect_handler.go | 3 +- retry_handler.go | 395 ++++++++++++++++++------------------- span_attributes.go | 34 ++++ 6 files changed, 405 insertions(+), 380 deletions(-) create mode 100644 span_attributes.go diff --git a/CHANGELOG.md b/CHANGELOG.md index 03feaf4..f8651bb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,17 +7,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] -## [1.4.7] - 2024-12-13 - -### Changed - -- Updated HTTP span attributes to comply with updated OpenTelemetry semantic conventions. [#182](https://github.com/microsoft/kiota-http-go/issues/182) - ## [1.4.6] - 2024-12-13 ### Changed - Fixed a bug where headers inspection handler would fail upon receiving an error. +- Updated HTTP span attributes to comply with updated OpenTelemetry semantic conventions. [#182](https://github.com/microsoft/kiota-http-go/issues/182) ## [1.4.5] - 2024-09-03 diff --git a/compression_handler.go b/compression_handler.go index fec1eb5..e33794c 100644 --- a/compression_handler.go +++ b/compression_handler.go @@ -1,162 +1,161 @@ -package nethttplibrary - -import ( - "bytes" - "compress/gzip" - "io" - "net/http" - "strings" - - abstractions "github.com/microsoft/kiota-abstractions-go" - "go.opentelemetry.io/otel" - "go.opentelemetry.io/otel/attribute" - semconv "go.opentelemetry.io/otel/semconv/v1.24.0" - "go.opentelemetry.io/otel/trace" -) - -// CompressionHandler represents a compression middleware -type CompressionHandler struct { - options CompressionOptions -} - -// CompressionOptions is a configuration object for the CompressionHandler middleware -type CompressionOptions struct { - enableCompression bool -} - -type compression interface { - abstractions.RequestOption - ShouldCompress() bool -} - -var compressKey = abstractions.RequestOptionKey{Key: "CompressionHandler"} - -// NewCompressionHandler creates an instance of a compression middleware -func NewCompressionHandler() *CompressionHandler { - options := NewCompressionOptions(true) - return NewCompressionHandlerWithOptions(options) -} - -// NewCompressionHandlerWithOptions creates an instance of the compression middleware with -// specified configurations. -func NewCompressionHandlerWithOptions(option CompressionOptions) *CompressionHandler { - return &CompressionHandler{options: option} -} - -// NewCompressionOptions creates a configuration object for the CompressionHandler -func NewCompressionOptions(enableCompression bool) CompressionOptions { - return CompressionOptions{enableCompression: enableCompression} -} - -// GetKey returns CompressionOptions unique name in context object -func (o CompressionOptions) GetKey() abstractions.RequestOptionKey { - return compressKey -} - -// ShouldCompress reads compression setting form CompressionOptions -func (o CompressionOptions) ShouldCompress() bool { - return o.enableCompression -} - -// Intercept is invoked by the middleware pipeline to either move the request/response -// to the next middleware in the pipeline -func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, req *http.Request) (*http.Response, error) { - reqOption, ok := req.Context().Value(compressKey).(compression) - if !ok { - reqOption = c.options - } - - obsOptions := GetObservabilityOptionsFromRequest(req) - ctx := req.Context() - var span trace.Span - if obsOptions != nil { - ctx, span = otel.GetTracerProvider().Tracer(obsOptions.GetTracerInstrumentationName()).Start(ctx, "CompressionHandler_Intercept") - span.SetAttributes(attribute.Bool("com.microsoft.kiota.handler.compression.enable", true)) - defer span.End() - req = req.WithContext(ctx) - } - - if !reqOption.ShouldCompress() || contentRangeBytesIsPresent(req.Header) || contentEncodingIsPresent(req.Header) || req.Body == nil { - return pipeline.Next(req, middlewareIndex) - } - if span != nil { - span.SetAttributes(attribute.Bool("http.request_body_compressed", true)) - } - - unCompressedBody, err := io.ReadAll(req.Body) - unCompressedContentLength := req.ContentLength - if err != nil { - if span != nil { - span.RecordError(err) - } - return nil, err - } - - compressedBody, size, err := compressReqBody(unCompressedBody) - if err != nil { - if span != nil { - span.RecordError(err) - } - return nil, err - } - - req.Header.Set("Content-Encoding", "gzip") - req.Body = compressedBody - req.ContentLength = int64(size) - - if span != nil { - span.SetAttributes(semconv.HTTPRequestBodySize(int(req.ContentLength))) - } - - // Sending request with compressed body - resp, err := pipeline.Next(req, middlewareIndex) - if err != nil { - return nil, err - } - - // If response has status 415 retry request with uncompressed body - if resp.StatusCode == 415 { - delete(req.Header, "Content-Encoding") - req.Body = io.NopCloser(bytes.NewBuffer(unCompressedBody)) - req.ContentLength = unCompressedContentLength - - if span != nil { - span.SetAttributes(semconv.HTTPRequestBodySize(int(req.ContentLength)), - semconv.HTTPResponseStatusCode(415)) - } - - return pipeline.Next(req, middlewareIndex) - } - - return resp, nil -} - -func contentRangeBytesIsPresent(header http.Header) bool { - contentRanges, _ := header["Content-Range"] - for _, contentRange := range contentRanges { - if strings.Contains(strings.ToLower(contentRange), "bytes") { - return true - } - } - return false -} - -func contentEncodingIsPresent(header http.Header) bool { - _, ok := header["Content-Encoding"] - return ok -} - -func compressReqBody(reqBody []byte) (io.ReadSeekCloser, int, error) { - var buffer bytes.Buffer - gzipWriter := gzip.NewWriter(&buffer) - if _, err := gzipWriter.Write(reqBody); err != nil { - return nil, 0, err - } - - if err := gzipWriter.Close(); err != nil { - return nil, 0, err - } - - reader := bytes.NewReader(buffer.Bytes()) - return NopCloser(reader), buffer.Len(), nil -} +package nethttplibrary + +import ( + "bytes" + "compress/gzip" + "io" + "net/http" + "strings" + + abstractions "github.com/microsoft/kiota-abstractions-go" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/trace" +) + +// CompressionHandler represents a compression middleware +type CompressionHandler struct { + options CompressionOptions +} + +// CompressionOptions is a configuration object for the CompressionHandler middleware +type CompressionOptions struct { + enableCompression bool +} + +type compression interface { + abstractions.RequestOption + ShouldCompress() bool +} + +var compressKey = abstractions.RequestOptionKey{Key: "CompressionHandler"} + +// NewCompressionHandler creates an instance of a compression middleware +func NewCompressionHandler() *CompressionHandler { + options := NewCompressionOptions(true) + return NewCompressionHandlerWithOptions(options) +} + +// NewCompressionHandlerWithOptions creates an instance of the compression middleware with +// specified configurations. +func NewCompressionHandlerWithOptions(option CompressionOptions) *CompressionHandler { + return &CompressionHandler{options: option} +} + +// NewCompressionOptions creates a configuration object for the CompressionHandler +func NewCompressionOptions(enableCompression bool) CompressionOptions { + return CompressionOptions{enableCompression: enableCompression} +} + +// GetKey returns CompressionOptions unique name in context object +func (o CompressionOptions) GetKey() abstractions.RequestOptionKey { + return compressKey +} + +// ShouldCompress reads compression setting form CompressionOptions +func (o CompressionOptions) ShouldCompress() bool { + return o.enableCompression +} + +// Intercept is invoked by the middleware pipeline to either move the request/response +// to the next middleware in the pipeline +func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, req *http.Request) (*http.Response, error) { + reqOption, ok := req.Context().Value(compressKey).(compression) + if !ok { + reqOption = c.options + } + + obsOptions := GetObservabilityOptionsFromRequest(req) + ctx := req.Context() + var span trace.Span + if obsOptions != nil { + ctx, span = otel.GetTracerProvider().Tracer(obsOptions.GetTracerInstrumentationName()).Start(ctx, "CompressionHandler_Intercept") + span.SetAttributes(attribute.Bool("com.microsoft.kiota.handler.compression.enable", true)) + defer span.End() + req = req.WithContext(ctx) + } + + if !reqOption.ShouldCompress() || contentRangeBytesIsPresent(req.Header) || contentEncodingIsPresent(req.Header) || req.Body == nil { + return pipeline.Next(req, middlewareIndex) + } + if span != nil { + span.SetAttributes(attribute.Bool("http.request_body_compressed", true)) + } + + unCompressedBody, err := io.ReadAll(req.Body) + unCompressedContentLength := req.ContentLength + if err != nil { + if span != nil { + span.RecordError(err) + } + return nil, err + } + + compressedBody, size, err := compressReqBody(unCompressedBody) + if err != nil { + if span != nil { + span.RecordError(err) + } + return nil, err + } + + req.Header.Set("Content-Encoding", "gzip") + req.Body = compressedBody + req.ContentLength = int64(size) + + if span != nil { + span.SetAttributes(HttpRequestBodySizeAttribute.Int(int(req.ContentLength))) + } + + // Sending request with compressed body + resp, err := pipeline.Next(req, middlewareIndex) + if err != nil { + return nil, err + } + + // If response has status 415 retry request with uncompressed body + if resp.StatusCode == 415 { + delete(req.Header, "Content-Encoding") + req.Body = io.NopCloser(bytes.NewBuffer(unCompressedBody)) + req.ContentLength = unCompressedContentLength + + if span != nil { + span.SetAttributes(HttpRequestBodySizeAttribute.Int(int(req.ContentLength)), + HttpResponseStatusCodeAttribute.Int(415)) + } + + return pipeline.Next(req, middlewareIndex) + } + + return resp, nil +} + +func contentRangeBytesIsPresent(header http.Header) bool { + contentRanges, _ := header["Content-Range"] + for _, contentRange := range contentRanges { + if strings.Contains(strings.ToLower(contentRange), "bytes") { + return true + } + } + return false +} + +func contentEncodingIsPresent(header http.Header) bool { + _, ok := header["Content-Encoding"] + return ok +} + +func compressReqBody(reqBody []byte) (io.ReadSeekCloser, int, error) { + var buffer bytes.Buffer + gzipWriter := gzip.NewWriter(&buffer) + if _, err := gzipWriter.Write(reqBody); err != nil { + return nil, 0, err + } + + if err := gzipWriter.Close(); err != nil { + return nil, 0, err + } + + reader := bytes.NewReader(buffer.Bytes()) + return NopCloser(reader), buffer.Len(), nil +} diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index 23a025f..68eb014 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -18,7 +18,6 @@ import ( "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" "go.opentelemetry.io/otel/codes" - semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -149,15 +148,15 @@ func (a *NetHttpRequestAdapter) getHttpResponseMessage(ctx context.Context, requ contentLenHeader := response.Header.Get("Content-Length") if contentLenHeader != "" { contentLen, _ := strconv.Atoi(contentLenHeader) - spanForAttributes.SetAttributes(semconv.HTTPResponseBodySize(contentLen)) + spanForAttributes.SetAttributes(HttpResponseBodySizeAttribute.Int(contentLen)) } contentTypeHeader := response.Header.Get("Content-Type") if contentTypeHeader != "" { - spanForAttributes.SetAttributes(attribute.String("http.response.header.content-type", contentTypeHeader)) + spanForAttributes.SetAttributes(HttpResponseHeaderContentTypeAttribute.String(contentTypeHeader)) } spanForAttributes.SetAttributes( - semconv.HTTPResponseStatusCode(response.StatusCode), - semconv.NetworkProtocolName(response.Proto), + HttpResponseStatusCodeAttribute.Int(response.StatusCode), + NetworkProtocolNameAttribute.String(response.Proto), ) } return a.retryCAEResponseIfRequired(ctx, response, requestInfo, claims, spanForAttributes) @@ -178,7 +177,7 @@ func (a *NetHttpRequestAdapter) retryCAEResponseIfRequired(ctx context.Context, authenticateHeaderVal := response.Header.Get("WWW-Authenticate") if authenticateHeaderVal != "" && reBearer.Match([]byte(authenticateHeaderVal)) { span.AddEvent(AuthenticateChallengedEventKey) - spanForAttributes.SetAttributes(semconv.HTTPRequestResendCount(1)) + spanForAttributes.SetAttributes(HttpRequestResendCountAttribute.Int(1)) responseClaims := "" parametersRaw := string(reBearer.ReplaceAll([]byte(authenticateHeaderVal), []byte(""))) parameters := strings.Split(parametersRaw, ",") @@ -254,19 +253,19 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con if spanForAttributes == nil { spanForAttributes = span } - spanForAttributes.SetAttributes(semconv.HTTPRequestMethodKey.String(requestInfo.Method.String())) + spanForAttributes.SetAttributes(HttpRequestMethodAttribute.String(requestInfo.Method.String())) uri, err := requestInfo.GetUri() if err != nil { spanForAttributes.RecordError(err) return nil, err } spanForAttributes.SetAttributes( - semconv.ServerAddress(uri.Scheme), - semconv.URLScheme(uri.Host), + ServerAddressAttribute.String(uri.Scheme), + UrlSchemeAttribute.String(uri.Host), ) if a.observabilityOptions.IncludeEUIIAttributes { - spanForAttributes.SetAttributes(semconv.URLFull(uri.String())) + spanForAttributes.SetAttributes(UrlFullAttribute.String(uri.String())) } request, err := nethttp.NewRequestWithContext(ctx, requestInfo.Method.String(), uri.String(), nil) @@ -298,7 +297,7 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con contentLenVal, _ := strconv.Atoi(request.Header.Get("Content-Length")) request.ContentLength = int64(contentLenVal) spanForAttributes.SetAttributes( - semconv.HTTPRequestBodySize(contentLenVal), + HttpRequestBodySizeAttribute.Int(contentLenVal), ) } } @@ -314,7 +313,7 @@ func (a *NetHttpRequestAdapter) startTracingSpan(ctx context.Context, requestInf decodedUriTemplate := decodeUriEncodedString(requestInfo.UrlTemplate, []byte{'-', '.', '~', '$'}) telemetryPathValue := queryParametersCleanupRegex.ReplaceAll([]byte(decodedUriTemplate), []byte("")) ctx, span := otel.GetTracerProvider().Tracer(a.observabilityOptions.GetTracerInstrumentationName()).Start(ctx, methodName+" - "+string(telemetryPathValue)) - span.SetAttributes(attribute.String("url.uri_template", decodedUriTemplate)) + span.SetAttributes(UrlUriSchemeAttribute.String(decodedUriTemplate)) return ctx, span } diff --git a/redirect_handler.go b/redirect_handler.go index eb71059..c7fbeac 100644 --- a/redirect_handler.go +++ b/redirect_handler.go @@ -11,7 +11,6 @@ import ( abs "github.com/microsoft/kiota-abstractions-go" "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" - semconv "go.opentelemetry.io/otel/semconv/v1.24.0" "go.opentelemetry.io/otel/trace" ) @@ -121,7 +120,7 @@ func (middleware RedirectHandler) redirectRequest(ctx context.Context, pipeline if observabilityName != "" { ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RedirectHandler_Intercept - redirect "+fmt.Sprint(redirectCount)) span.SetAttributes(attribute.Int("com.microsoft.kiota.handler.redirect.count", redirectCount), - semconv.HTTPResponseStatusCode(response.StatusCode), + HttpResponseStatusCodeAttribute.Int(response.StatusCode), ) defer span.End() redirectRequest = redirectRequest.WithContext(ctx) diff --git a/retry_handler.go b/retry_handler.go index 2b65f36..086779b 100644 --- a/retry_handler.go +++ b/retry_handler.go @@ -1,198 +1,197 @@ -package nethttplibrary - -import ( - "context" - "fmt" - "io" - "math" - nethttp "net/http" - "strconv" - "time" - - abs "github.com/microsoft/kiota-abstractions-go" - "go.opentelemetry.io/otel" - "go.opentelemetry.io/otel/attribute" - semconv "go.opentelemetry.io/otel/semconv/v1.24.0" - "go.opentelemetry.io/otel/trace" -) - -// RetryHandler handles transient HTTP responses and retries the request given the retry options -type RetryHandler struct { - // default options to use when evaluating the response - options RetryHandlerOptions -} - -// NewRetryHandler creates a new RetryHandler with default options -func NewRetryHandler() *RetryHandler { - return NewRetryHandlerWithOptions(RetryHandlerOptions{ - ShouldRetry: func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool { - return true - }, - }) -} - -// NewRetryHandlerWithOptions creates a new RetryHandler with the given options -func NewRetryHandlerWithOptions(options RetryHandlerOptions) *RetryHandler { - return &RetryHandler{options: options} -} - -const defaultMaxRetries = 3 -const absoluteMaxRetries = 10 -const defaultDelaySeconds = 3 -const absoluteMaxDelaySeconds = 180 - -// RetryHandlerOptions to apply when evaluating the response for retrial -type RetryHandlerOptions struct { - // Callback to determine if the request should be retried - ShouldRetry func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool - // The maximum number of times a request can be retried - MaxRetries int - // The delay in seconds between retries - DelaySeconds int -} - -type retryHandlerOptionsInt interface { - abs.RequestOption - GetShouldRetry() func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool - GetDelaySeconds() int - GetMaxRetries() int -} - -var retryKeyValue = abs.RequestOptionKey{ - Key: "RetryHandler", -} - -// GetKey returns the key value to be used when the option is added to the request context -func (options *RetryHandlerOptions) GetKey() abs.RequestOptionKey { - return retryKeyValue -} - -// GetShouldRetry returns the should retry callback function which evaluates the response for retrial -func (options *RetryHandlerOptions) GetShouldRetry() func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool { - return options.ShouldRetry -} - -// GetDelaySeconds returns the delays in seconds between retries -func (options *RetryHandlerOptions) GetDelaySeconds() int { - if options.DelaySeconds < 1 { - return defaultDelaySeconds - } else if options.DelaySeconds > absoluteMaxDelaySeconds { - return absoluteMaxDelaySeconds - } else { - return options.DelaySeconds - } -} - -// GetMaxRetries returns the maximum number of times a request can be retried -func (options *RetryHandlerOptions) GetMaxRetries() int { - if options.MaxRetries < 1 { - return defaultMaxRetries - } else if options.MaxRetries > absoluteMaxRetries { - return absoluteMaxRetries - } else { - return options.MaxRetries - } -} - -const retryAttemptHeader = "Retry-Attempt" -const retryAfterHeader = "Retry-After" - -const tooManyRequests = 429 -const serviceUnavailable = 503 -const gatewayTimeout = 504 - -// Intercept implements the interface and evaluates whether to retry a failed request. -func (middleware RetryHandler) Intercept(pipeline Pipeline, middlewareIndex int, req *nethttp.Request) (*nethttp.Response, error) { - obsOptions := GetObservabilityOptionsFromRequest(req) - ctx := req.Context() - var span trace.Span - var observabilityName string - if obsOptions != nil { - observabilityName = obsOptions.GetTracerInstrumentationName() - ctx, span = otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept") - span.SetAttributes(attribute.Bool("com.microsoft.kiota.handler.retry.enable", true)) - defer span.End() - req = req.WithContext(ctx) - } - response, err := pipeline.Next(req, middlewareIndex) - if err != nil { - return response, err - } - reqOption, ok := req.Context().Value(retryKeyValue).(retryHandlerOptionsInt) - if !ok { - reqOption = &middleware.options - } - return middleware.retryRequest(ctx, pipeline, middlewareIndex, reqOption, req, response, 0, 0, observabilityName) -} - -func (middleware RetryHandler) retryRequest(ctx context.Context, pipeline Pipeline, middlewareIndex int, options retryHandlerOptionsInt, req *nethttp.Request, resp *nethttp.Response, executionCount int, cumulativeDelay time.Duration, observabilityName string) (*nethttp.Response, error) { - if middleware.isRetriableErrorCode(resp.StatusCode) && - middleware.isRetriableRequest(req) && - executionCount < options.GetMaxRetries() && - cumulativeDelay < time.Duration(absoluteMaxDelaySeconds)*time.Second && - options.GetShouldRetry()(cumulativeDelay, executionCount, req, resp) { - executionCount++ - delay := middleware.getRetryDelay(req, resp, options, executionCount) - cumulativeDelay += delay - req.Header.Set(retryAttemptHeader, strconv.Itoa(executionCount)) - if req.Body != nil { - s, ok := req.Body.(io.Seeker) - if ok { - s.Seek(0, io.SeekStart) - } - } - if observabilityName != "" { - ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept - attempt "+fmt.Sprint(executionCount)) - span.SetAttributes(attribute.Int("http.request.resend_count", executionCount), - - semconv.HTTPResponseStatusCode(resp.StatusCode), - attribute.Float64("http.request.resend_delay", delay.Seconds()), - ) - defer span.End() - req = req.WithContext(ctx) - } - t := time.NewTimer(delay) - select { - case <-ctx.Done(): - // Return without retrying if the context was cancelled. - return nil, ctx.Err() - - // Leaving this case empty causes it to exit the switch-block. - case <-t.C: - } - response, err := pipeline.Next(req, middlewareIndex) - if err != nil { - return response, err - } - return middleware.retryRequest(ctx, pipeline, middlewareIndex, options, req, response, executionCount, cumulativeDelay, observabilityName) - } - return resp, nil -} - -func (middleware RetryHandler) isRetriableErrorCode(code int) bool { - return code == tooManyRequests || code == serviceUnavailable || code == gatewayTimeout -} -func (middleware RetryHandler) isRetriableRequest(req *nethttp.Request) bool { - isBodiedMethod := req.Method == "POST" || req.Method == "PUT" || req.Method == "PATCH" - if isBodiedMethod && req.Body != nil { - return req.ContentLength != -1 - } - return true -} - -func (middleware RetryHandler) getRetryDelay(req *nethttp.Request, resp *nethttp.Response, options retryHandlerOptionsInt, executionCount int) time.Duration { - retryAfter := resp.Header.Get(retryAfterHeader) - if retryAfter != "" { - retryAfterDelay, err := strconv.ParseFloat(retryAfter, 64) - if err == nil { - return time.Duration(retryAfterDelay) * time.Second - } - - // parse the header if it's a date - t, err := time.Parse(time.RFC1123, retryAfter) - if err == nil { - return t.Sub(time.Now()) - } - } - return time.Duration(math.Pow(float64(options.GetDelaySeconds()), float64(executionCount))) * time.Second -} +package nethttplibrary + +import ( + "context" + "fmt" + "io" + "math" + nethttp "net/http" + "strconv" + "time" + + abs "github.com/microsoft/kiota-abstractions-go" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/trace" +) + +// RetryHandler handles transient HTTP responses and retries the request given the retry options +type RetryHandler struct { + // default options to use when evaluating the response + options RetryHandlerOptions +} + +// NewRetryHandler creates a new RetryHandler with default options +func NewRetryHandler() *RetryHandler { + return NewRetryHandlerWithOptions(RetryHandlerOptions{ + ShouldRetry: func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool { + return true + }, + }) +} + +// NewRetryHandlerWithOptions creates a new RetryHandler with the given options +func NewRetryHandlerWithOptions(options RetryHandlerOptions) *RetryHandler { + return &RetryHandler{options: options} +} + +const defaultMaxRetries = 3 +const absoluteMaxRetries = 10 +const defaultDelaySeconds = 3 +const absoluteMaxDelaySeconds = 180 + +// RetryHandlerOptions to apply when evaluating the response for retrial +type RetryHandlerOptions struct { + // Callback to determine if the request should be retried + ShouldRetry func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool + // The maximum number of times a request can be retried + MaxRetries int + // The delay in seconds between retries + DelaySeconds int +} + +type retryHandlerOptionsInt interface { + abs.RequestOption + GetShouldRetry() func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool + GetDelaySeconds() int + GetMaxRetries() int +} + +var retryKeyValue = abs.RequestOptionKey{ + Key: "RetryHandler", +} + +// GetKey returns the key value to be used when the option is added to the request context +func (options *RetryHandlerOptions) GetKey() abs.RequestOptionKey { + return retryKeyValue +} + +// GetShouldRetry returns the should retry callback function which evaluates the response for retrial +func (options *RetryHandlerOptions) GetShouldRetry() func(delay time.Duration, executionCount int, request *nethttp.Request, response *nethttp.Response) bool { + return options.ShouldRetry +} + +// GetDelaySeconds returns the delays in seconds between retries +func (options *RetryHandlerOptions) GetDelaySeconds() int { + if options.DelaySeconds < 1 { + return defaultDelaySeconds + } else if options.DelaySeconds > absoluteMaxDelaySeconds { + return absoluteMaxDelaySeconds + } else { + return options.DelaySeconds + } +} + +// GetMaxRetries returns the maximum number of times a request can be retried +func (options *RetryHandlerOptions) GetMaxRetries() int { + if options.MaxRetries < 1 { + return defaultMaxRetries + } else if options.MaxRetries > absoluteMaxRetries { + return absoluteMaxRetries + } else { + return options.MaxRetries + } +} + +const retryAttemptHeader = "Retry-Attempt" +const retryAfterHeader = "Retry-After" + +const tooManyRequests = 429 +const serviceUnavailable = 503 +const gatewayTimeout = 504 + +// Intercept implements the interface and evaluates whether to retry a failed request. +func (middleware RetryHandler) Intercept(pipeline Pipeline, middlewareIndex int, req *nethttp.Request) (*nethttp.Response, error) { + obsOptions := GetObservabilityOptionsFromRequest(req) + ctx := req.Context() + var span trace.Span + var observabilityName string + if obsOptions != nil { + observabilityName = obsOptions.GetTracerInstrumentationName() + ctx, span = otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept") + span.SetAttributes(attribute.Bool("com.microsoft.kiota.handler.retry.enable", true)) + defer span.End() + req = req.WithContext(ctx) + } + response, err := pipeline.Next(req, middlewareIndex) + if err != nil { + return response, err + } + reqOption, ok := req.Context().Value(retryKeyValue).(retryHandlerOptionsInt) + if !ok { + reqOption = &middleware.options + } + return middleware.retryRequest(ctx, pipeline, middlewareIndex, reqOption, req, response, 0, 0, observabilityName) +} + +func (middleware RetryHandler) retryRequest(ctx context.Context, pipeline Pipeline, middlewareIndex int, options retryHandlerOptionsInt, req *nethttp.Request, resp *nethttp.Response, executionCount int, cumulativeDelay time.Duration, observabilityName string) (*nethttp.Response, error) { + if middleware.isRetriableErrorCode(resp.StatusCode) && + middleware.isRetriableRequest(req) && + executionCount < options.GetMaxRetries() && + cumulativeDelay < time.Duration(absoluteMaxDelaySeconds)*time.Second && + options.GetShouldRetry()(cumulativeDelay, executionCount, req, resp) { + executionCount++ + delay := middleware.getRetryDelay(req, resp, options, executionCount) + cumulativeDelay += delay + req.Header.Set(retryAttemptHeader, strconv.Itoa(executionCount)) + if req.Body != nil { + s, ok := req.Body.(io.Seeker) + if ok { + s.Seek(0, io.SeekStart) + } + } + if observabilityName != "" { + ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept - attempt "+fmt.Sprint(executionCount)) + span.SetAttributes(attribute.Int("http.request.resend_count", executionCount), + + HttpResponseStatusCodeAttribute.Int(resp.StatusCode), + attribute.Float64("http.request.resend_delay", delay.Seconds()), + ) + defer span.End() + req = req.WithContext(ctx) + } + t := time.NewTimer(delay) + select { + case <-ctx.Done(): + // Return without retrying if the context was cancelled. + return nil, ctx.Err() + + // Leaving this case empty causes it to exit the switch-block. + case <-t.C: + } + response, err := pipeline.Next(req, middlewareIndex) + if err != nil { + return response, err + } + return middleware.retryRequest(ctx, pipeline, middlewareIndex, options, req, response, executionCount, cumulativeDelay, observabilityName) + } + return resp, nil +} + +func (middleware RetryHandler) isRetriableErrorCode(code int) bool { + return code == tooManyRequests || code == serviceUnavailable || code == gatewayTimeout +} +func (middleware RetryHandler) isRetriableRequest(req *nethttp.Request) bool { + isBodiedMethod := req.Method == "POST" || req.Method == "PUT" || req.Method == "PATCH" + if isBodiedMethod && req.Body != nil { + return req.ContentLength != -1 + } + return true +} + +func (middleware RetryHandler) getRetryDelay(req *nethttp.Request, resp *nethttp.Response, options retryHandlerOptionsInt, executionCount int) time.Duration { + retryAfter := resp.Header.Get(retryAfterHeader) + if retryAfter != "" { + retryAfterDelay, err := strconv.ParseFloat(retryAfter, 64) + if err == nil { + return time.Duration(retryAfterDelay) * time.Second + } + + // parse the header if it's a date + t, err := time.Parse(time.RFC1123, retryAfter) + if err == nil { + return t.Sub(time.Now()) + } + } + return time.Duration(math.Pow(float64(options.GetDelaySeconds()), float64(executionCount))) * time.Second +} diff --git a/span_attributes.go b/span_attributes.go new file mode 100644 index 0000000..39beaa9 --- /dev/null +++ b/span_attributes.go @@ -0,0 +1,34 @@ +package nethttplibrary + +import "go.opentelemetry.io/otel/attribute" + +// HTTP Request attributes +const ( + HttpRequestBodySizeAttribute = attribute.Key("http.request.body.size") + HttpRequestResendCountAttribute = attribute.Key("http.request.resend_count") + HttpRequestMethodAttribute = attribute.Key("http.request.method") +) + +// HTTP Response attributes +const ( + HttpResponseBodySizeAttribute = attribute.Key("http.response.body.size") + HttpResponseHeaderContentTypeAttribute = attribute.Key("http.response.header.content_type") + HttpResponseStatusCodeAttribute = attribute.Key("http.response.status_code") +) + +// Network attributes +const ( + NetworkProtocolNameAttribute = attribute.Key("network.protocol.name") +) + +// Server attributes +const ( + ServerAddressAttribute = attribute.Key("server.address") +) + +// URL attributes +const ( + UrlFullAttribute = attribute.Key("url.full") + UrlSchemeAttribute = attribute.Key("url.scheme") + UrlUriSchemeAttribute = attribute.Key("url.uri_scheme") +) From 4b145148508da15ca317acebdbfe4f7fd11ca4bd Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Fri, 13 Dec 2024 12:11:01 -0500 Subject: [PATCH 4/7] Apply suggestions from code review --- nethttp_request_adapter.go | 2 +- span_attributes.go | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index 68eb014..94e4e06 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -313,7 +313,7 @@ func (a *NetHttpRequestAdapter) startTracingSpan(ctx context.Context, requestInf decodedUriTemplate := decodeUriEncodedString(requestInfo.UrlTemplate, []byte{'-', '.', '~', '$'}) telemetryPathValue := queryParametersCleanupRegex.ReplaceAll([]byte(decodedUriTemplate), []byte("")) ctx, span := otel.GetTracerProvider().Tracer(a.observabilityOptions.GetTracerInstrumentationName()).Start(ctx, methodName+" - "+string(telemetryPathValue)) - span.SetAttributes(UrlUriSchemeAttribute.String(decodedUriTemplate)) + span.SetAttributes(UrlUriTemplateAttribute.String(decodedUriTemplate)) return ctx, span } diff --git a/span_attributes.go b/span_attributes.go index 39beaa9..23a251a 100644 --- a/span_attributes.go +++ b/span_attributes.go @@ -12,7 +12,7 @@ const ( // HTTP Response attributes const ( HttpResponseBodySizeAttribute = attribute.Key("http.response.body.size") - HttpResponseHeaderContentTypeAttribute = attribute.Key("http.response.header.content_type") + HttpResponseHeaderContentTypeAttribute = attribute.Key("http.response.header.content-type") HttpResponseStatusCodeAttribute = attribute.Key("http.response.status_code") ) @@ -30,5 +30,5 @@ const ( const ( UrlFullAttribute = attribute.Key("url.full") UrlSchemeAttribute = attribute.Key("url.scheme") - UrlUriSchemeAttribute = attribute.Key("url.uri_scheme") + UrlUriTemplateAttribute = attribute.Key("url.uri_template") ) From 3692bb2fe1636ac9b500d9b61d7031c7a3f83020 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Fri, 13 Dec 2024 12:12:55 -0500 Subject: [PATCH 5/7] chore: bumps patch version Signed-off-by: Vincent Biret --- CHANGELOG.md | 7 ++++++- user_agent_handler.go | 2 +- 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f8651bb..03feaf4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,12 +7,17 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +## [1.4.7] - 2024-12-13 + +### Changed + +- Updated HTTP span attributes to comply with updated OpenTelemetry semantic conventions. [#182](https://github.com/microsoft/kiota-http-go/issues/182) + ## [1.4.6] - 2024-12-13 ### Changed - Fixed a bug where headers inspection handler would fail upon receiving an error. -- Updated HTTP span attributes to comply with updated OpenTelemetry semantic conventions. [#182](https://github.com/microsoft/kiota-http-go/issues/182) ## [1.4.5] - 2024-09-03 diff --git a/user_agent_handler.go b/user_agent_handler.go index 51af61b..620ab2d 100644 --- a/user_agent_handler.go +++ b/user_agent_handler.go @@ -42,7 +42,7 @@ func NewUserAgentHandlerOptions() *UserAgentHandlerOptions { return &UserAgentHandlerOptions{ Enabled: true, ProductName: "kiota-go", - ProductVersion: "1.4.6", + ProductVersion: "1.4.7", } } From e1f0d35fa6555cbd5598b94d5d55e47fd1f04740 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Fri, 13 Dec 2024 12:15:09 -0500 Subject: [PATCH 6/7] fix: normalizes request header constant Signed-off-by: Vincent Biret --- nethttp_request_adapter.go | 2 +- span_attributes.go | 11 ++++++----- 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index 94e4e06..e0a19fe 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -290,7 +290,7 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con } if request.Header.Get("Content-Type") != "" { spanForAttributes.SetAttributes( - attribute.String("http.request.header.content-type", request.Header.Get("Content-Type")), + HttpRequestHeaderContentTypeAttribute.String(request.Header.Get("Content-Type")), ) } if request.Header.Get("Content-Length") != "" { diff --git a/span_attributes.go b/span_attributes.go index 23a251a..c0d8866 100644 --- a/span_attributes.go +++ b/span_attributes.go @@ -4,9 +4,10 @@ import "go.opentelemetry.io/otel/attribute" // HTTP Request attributes const ( - HttpRequestBodySizeAttribute = attribute.Key("http.request.body.size") - HttpRequestResendCountAttribute = attribute.Key("http.request.resend_count") - HttpRequestMethodAttribute = attribute.Key("http.request.method") + HttpRequestBodySizeAttribute = attribute.Key("http.request.body.size") + HttpRequestResendCountAttribute = attribute.Key("http.request.resend_count") + HttpRequestMethodAttribute = attribute.Key("http.request.method") + HttpRequestHeaderContentTypeAttribute = attribute.Key("http.request.header.content-type") ) // HTTP Response attributes @@ -28,7 +29,7 @@ const ( // URL attributes const ( - UrlFullAttribute = attribute.Key("url.full") - UrlSchemeAttribute = attribute.Key("url.scheme") + UrlFullAttribute = attribute.Key("url.full") + UrlSchemeAttribute = attribute.Key("url.scheme") UrlUriTemplateAttribute = attribute.Key("url.uri_template") ) From 9cce1d810127010bf1a8535aeaea25f602c743c1 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Fri, 13 Dec 2024 12:16:05 -0500 Subject: [PATCH 7/7] fix: makes all constants private Signed-off-by: Vincent Biret --- compression_handler.go | 6 +++--- nethttp_request_adapter.go | 24 ++++++++++++------------ redirect_handler.go | 2 +- retry_handler.go | 2 +- span_attributes.go | 24 ++++++++++++------------ 5 files changed, 29 insertions(+), 29 deletions(-) diff --git a/compression_handler.go b/compression_handler.go index e33794c..526b158 100644 --- a/compression_handler.go +++ b/compression_handler.go @@ -104,7 +104,7 @@ func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, r req.ContentLength = int64(size) if span != nil { - span.SetAttributes(HttpRequestBodySizeAttribute.Int(int(req.ContentLength))) + span.SetAttributes(httpRequestBodySizeAttribute.Int(int(req.ContentLength))) } // Sending request with compressed body @@ -120,8 +120,8 @@ func (c *CompressionHandler) Intercept(pipeline Pipeline, middlewareIndex int, r req.ContentLength = unCompressedContentLength if span != nil { - span.SetAttributes(HttpRequestBodySizeAttribute.Int(int(req.ContentLength)), - HttpResponseStatusCodeAttribute.Int(415)) + span.SetAttributes(httpRequestBodySizeAttribute.Int(int(req.ContentLength)), + httpResponseStatusCodeAttribute.Int(415)) } return pipeline.Next(req, middlewareIndex) diff --git a/nethttp_request_adapter.go b/nethttp_request_adapter.go index e0a19fe..4bf170a 100644 --- a/nethttp_request_adapter.go +++ b/nethttp_request_adapter.go @@ -148,15 +148,15 @@ func (a *NetHttpRequestAdapter) getHttpResponseMessage(ctx context.Context, requ contentLenHeader := response.Header.Get("Content-Length") if contentLenHeader != "" { contentLen, _ := strconv.Atoi(contentLenHeader) - spanForAttributes.SetAttributes(HttpResponseBodySizeAttribute.Int(contentLen)) + spanForAttributes.SetAttributes(httpResponseBodySizeAttribute.Int(contentLen)) } contentTypeHeader := response.Header.Get("Content-Type") if contentTypeHeader != "" { - spanForAttributes.SetAttributes(HttpResponseHeaderContentTypeAttribute.String(contentTypeHeader)) + spanForAttributes.SetAttributes(httpResponseHeaderContentTypeAttribute.String(contentTypeHeader)) } spanForAttributes.SetAttributes( - HttpResponseStatusCodeAttribute.Int(response.StatusCode), - NetworkProtocolNameAttribute.String(response.Proto), + httpResponseStatusCodeAttribute.Int(response.StatusCode), + networkProtocolNameAttribute.String(response.Proto), ) } return a.retryCAEResponseIfRequired(ctx, response, requestInfo, claims, spanForAttributes) @@ -177,7 +177,7 @@ func (a *NetHttpRequestAdapter) retryCAEResponseIfRequired(ctx context.Context, authenticateHeaderVal := response.Header.Get("WWW-Authenticate") if authenticateHeaderVal != "" && reBearer.Match([]byte(authenticateHeaderVal)) { span.AddEvent(AuthenticateChallengedEventKey) - spanForAttributes.SetAttributes(HttpRequestResendCountAttribute.Int(1)) + spanForAttributes.SetAttributes(httpRequestResendCountAttribute.Int(1)) responseClaims := "" parametersRaw := string(reBearer.ReplaceAll([]byte(authenticateHeaderVal), []byte(""))) parameters := strings.Split(parametersRaw, ",") @@ -253,19 +253,19 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con if spanForAttributes == nil { spanForAttributes = span } - spanForAttributes.SetAttributes(HttpRequestMethodAttribute.String(requestInfo.Method.String())) + spanForAttributes.SetAttributes(httpRequestMethodAttribute.String(requestInfo.Method.String())) uri, err := requestInfo.GetUri() if err != nil { spanForAttributes.RecordError(err) return nil, err } spanForAttributes.SetAttributes( - ServerAddressAttribute.String(uri.Scheme), - UrlSchemeAttribute.String(uri.Host), + serverAddressAttribute.String(uri.Scheme), + urlSchemeAttribute.String(uri.Host), ) if a.observabilityOptions.IncludeEUIIAttributes { - spanForAttributes.SetAttributes(UrlFullAttribute.String(uri.String())) + spanForAttributes.SetAttributes(urlFullAttribute.String(uri.String())) } request, err := nethttp.NewRequestWithContext(ctx, requestInfo.Method.String(), uri.String(), nil) @@ -290,14 +290,14 @@ func (a *NetHttpRequestAdapter) getRequestFromRequestInformation(ctx context.Con } if request.Header.Get("Content-Type") != "" { spanForAttributes.SetAttributes( - HttpRequestHeaderContentTypeAttribute.String(request.Header.Get("Content-Type")), + httpRequestHeaderContentTypeAttribute.String(request.Header.Get("Content-Type")), ) } if request.Header.Get("Content-Length") != "" { contentLenVal, _ := strconv.Atoi(request.Header.Get("Content-Length")) request.ContentLength = int64(contentLenVal) spanForAttributes.SetAttributes( - HttpRequestBodySizeAttribute.Int(contentLenVal), + httpRequestBodySizeAttribute.Int(contentLenVal), ) } } @@ -313,7 +313,7 @@ func (a *NetHttpRequestAdapter) startTracingSpan(ctx context.Context, requestInf decodedUriTemplate := decodeUriEncodedString(requestInfo.UrlTemplate, []byte{'-', '.', '~', '$'}) telemetryPathValue := queryParametersCleanupRegex.ReplaceAll([]byte(decodedUriTemplate), []byte("")) ctx, span := otel.GetTracerProvider().Tracer(a.observabilityOptions.GetTracerInstrumentationName()).Start(ctx, methodName+" - "+string(telemetryPathValue)) - span.SetAttributes(UrlUriTemplateAttribute.String(decodedUriTemplate)) + span.SetAttributes(urlUriTemplateAttribute.String(decodedUriTemplate)) return ctx, span } diff --git a/redirect_handler.go b/redirect_handler.go index c7fbeac..f6f63bb 100644 --- a/redirect_handler.go +++ b/redirect_handler.go @@ -120,7 +120,7 @@ func (middleware RedirectHandler) redirectRequest(ctx context.Context, pipeline if observabilityName != "" { ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RedirectHandler_Intercept - redirect "+fmt.Sprint(redirectCount)) span.SetAttributes(attribute.Int("com.microsoft.kiota.handler.redirect.count", redirectCount), - HttpResponseStatusCodeAttribute.Int(response.StatusCode), + httpResponseStatusCodeAttribute.Int(response.StatusCode), ) defer span.End() redirectRequest = redirectRequest.WithContext(ctx) diff --git a/retry_handler.go b/retry_handler.go index 086779b..6273de7 100644 --- a/retry_handler.go +++ b/retry_handler.go @@ -144,7 +144,7 @@ func (middleware RetryHandler) retryRequest(ctx context.Context, pipeline Pipeli ctx, span := otel.GetTracerProvider().Tracer(observabilityName).Start(ctx, "RetryHandler_Intercept - attempt "+fmt.Sprint(executionCount)) span.SetAttributes(attribute.Int("http.request.resend_count", executionCount), - HttpResponseStatusCodeAttribute.Int(resp.StatusCode), + httpResponseStatusCodeAttribute.Int(resp.StatusCode), attribute.Float64("http.request.resend_delay", delay.Seconds()), ) defer span.End() diff --git a/span_attributes.go b/span_attributes.go index c0d8866..c91f341 100644 --- a/span_attributes.go +++ b/span_attributes.go @@ -4,32 +4,32 @@ import "go.opentelemetry.io/otel/attribute" // HTTP Request attributes const ( - HttpRequestBodySizeAttribute = attribute.Key("http.request.body.size") - HttpRequestResendCountAttribute = attribute.Key("http.request.resend_count") - HttpRequestMethodAttribute = attribute.Key("http.request.method") - HttpRequestHeaderContentTypeAttribute = attribute.Key("http.request.header.content-type") + httpRequestBodySizeAttribute = attribute.Key("http.request.body.size") + httpRequestResendCountAttribute = attribute.Key("http.request.resend_count") + httpRequestMethodAttribute = attribute.Key("http.request.method") + httpRequestHeaderContentTypeAttribute = attribute.Key("http.request.header.content-type") ) // HTTP Response attributes const ( - HttpResponseBodySizeAttribute = attribute.Key("http.response.body.size") - HttpResponseHeaderContentTypeAttribute = attribute.Key("http.response.header.content-type") - HttpResponseStatusCodeAttribute = attribute.Key("http.response.status_code") + httpResponseBodySizeAttribute = attribute.Key("http.response.body.size") + httpResponseHeaderContentTypeAttribute = attribute.Key("http.response.header.content-type") + httpResponseStatusCodeAttribute = attribute.Key("http.response.status_code") ) // Network attributes const ( - NetworkProtocolNameAttribute = attribute.Key("network.protocol.name") + networkProtocolNameAttribute = attribute.Key("network.protocol.name") ) // Server attributes const ( - ServerAddressAttribute = attribute.Key("server.address") + serverAddressAttribute = attribute.Key("server.address") ) // URL attributes const ( - UrlFullAttribute = attribute.Key("url.full") - UrlSchemeAttribute = attribute.Key("url.scheme") - UrlUriTemplateAttribute = attribute.Key("url.uri_template") + urlFullAttribute = attribute.Key("url.full") + urlSchemeAttribute = attribute.Key("url.scheme") + urlUriTemplateAttribute = attribute.Key("url.uri_template") )