From c4ba1117bdfb721f25548c1f4406465846c8241f Mon Sep 17 00:00:00 2001 From: Wouter Dullaert Date: Wed, 21 Sep 2022 17:48:02 +0200 Subject: [PATCH 1/4] Implement specification compliant trace status handling --- CHANGELOG.md | 5 +++ sdk/trace/span.go | 7 ++-- sdk/trace/span_test.go | 76 ++++++++++++++++++++++++++++++++++++++++++ trace/trace.go | 4 +-- 4 files changed, 88 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 03ea381e10f..2719ac11915 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,6 +13,11 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm - The metric portion of the OpenCensus bridge (`go.opentelemetry.io/otel/bridge/opencensus`) has been reintroduced. (#3192) - The OpenCensus bridge example (`go.opentelemetry.io/otel/example/opencensus`) has been reintroduced. (#3206) +### Changed + +- `span.SetStatus` has been updated to comply with the OpenTelemetry specification. + Calls that lower the status are now noops. (#3214) + ## [0.32.0] Revised Metric SDK (Alpha) - 2022-09-18 ### Changed diff --git a/sdk/trace/span.go b/sdk/trace/span.go index 449cf6c2552..9760923f702 100644 --- a/sdk/trace/span.go +++ b/sdk/trace/span.go @@ -189,15 +189,18 @@ func (s *recordingSpan) SetStatus(code codes.Code, description string) { if !s.IsRecording() { return } + s.mu.Lock() + defer s.mu.Unlock() + if s.status.Code > code { + return + } status := Status{Code: code} if code == codes.Error { status.Description = description } - s.mu.Lock() s.status = status - s.mu.Unlock() } // SetAttributes sets attributes of this span. diff --git a/sdk/trace/span_test.go b/sdk/trace/span_test.go index 2441defae71..20148a78702 100644 --- a/sdk/trace/span_test.go +++ b/sdk/trace/span_test.go @@ -22,8 +22,84 @@ import ( "github.com/stretchr/testify/assert" "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/codes" ) +func TestSetStatus(t *testing.T) { + tests := []struct { + name string + span recordingSpan + code codes.Code + description string + expected Status + }{ + { + "Error and description should overwrite Unset", + recordingSpan{}, + codes.Error, + "description", + Status{Code: codes.Error, Description: "description"}, + }, + { + "Ok should overwrite Unset and ignore description", + recordingSpan{}, + codes.Ok, + "description", + Status{Code: codes.Ok}, + }, + { + "Error and description should return error and overwrite description", + recordingSpan{status: Status{Code: codes.Error, Description: "d1"}}, + codes.Error, + "d2", + Status{Code: codes.Error, Description: "d2"}, + }, + { + "Ok should overwrite error and remove description", + recordingSpan{status: Status{Code: codes.Error, Description: "d1"}}, + codes.Ok, + "d2", + Status{Code: codes.Ok}, + }, + { + "Error and description should be ignored when already Ok", + recordingSpan{status: Status{Code: codes.Ok}}, + codes.Error, + "d2", + Status{Code: codes.Ok}, + }, + { + "Ok should be noop when already Ok", + recordingSpan{status: Status{Code: codes.Ok}}, + codes.Ok, + "d2", + Status{Code: codes.Ok}, + }, + { + "Unset should be noop when already Ok", + recordingSpan{status: Status{Code: codes.Ok}}, + codes.Unset, + "d2", + Status{Code: codes.Ok}, + }, + { + "Unset should be noop when already Error", + recordingSpan{status: Status{Code: codes.Error, Description: "d1"}}, + codes.Unset, + "d2", + Status{Code: codes.Error, Description: "d1"}, + }, + } + + for i := range tests { + tc := &tests[i] + t.Run(tc.name, func(t *testing.T) { + tc.span.SetStatus(tc.code, tc.description) + assert.Equal(t, tc.expected, tc.span.status) + }) + } +} + func TestTruncateAttr(t *testing.T) { const key = "key" diff --git a/trace/trace.go b/trace/trace.go index 97f3d83855b..4ff03e86d52 100644 --- a/trace/trace.go +++ b/trace/trace.go @@ -364,8 +364,8 @@ type Span interface { SpanContext() SpanContext // SetStatus sets the status of the Span in the form of a code and a - // description, overriding previous values set. The description is only - // included in a status when the code is for an error. + // description. The description is only included in a status when the code + // is for an error. SetStatus(code codes.Code, description string) // SetName sets the Span name. From cc0534f998c1777fdd445eb24d1e5bb0a05737e0 Mon Sep 17 00:00:00 2001 From: wdullaer Date: Thu, 22 Sep 2022 13:59:36 +0200 Subject: [PATCH 2/4] Update trace/trace.go Co-authored-by: Damien Mathieu <42@dmathieu.com> --- trace/trace.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/trace/trace.go b/trace/trace.go index 4ff03e86d52..1b5ffb00a1f 100644 --- a/trace/trace.go +++ b/trace/trace.go @@ -364,8 +364,9 @@ type Span interface { SpanContext() SpanContext // SetStatus sets the status of the Span in the form of a code and a - // description. The description is only included in a status when the code - // is for an error. + // description, provided the status hasn't already been set to a higher + // value before (OK > Error > Unset). The description is only included in a + // status when the code is for an error. SetStatus(code codes.Code, description string) // SetName sets the Span name. From 3802c6dcd9d5cc8cce218f975306882b5803ca67 Mon Sep 17 00:00:00 2001 From: wdullaer Date: Thu, 22 Sep 2022 17:42:40 +0200 Subject: [PATCH 3/4] Update CHANGELOG.md Co-authored-by: Anthony Mirabella --- CHANGELOG.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2719ac11915..1ad285cbd7f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,8 +15,7 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm ### Changed -- `span.SetStatus` has been updated to comply with the OpenTelemetry specification. - Calls that lower the status are now noops. (#3214) +- `span.SetStatus` has been updated such that calls that lower the status are now no-ops. (#3214) ## [0.32.0] Revised Metric SDK (Alpha) - 2022-09-18 From 50a1df934b0c100a4772b49ab5c864eae2f06ef0 Mon Sep 17 00:00:00 2001 From: Wouter Dullaert Date: Fri, 23 Sep 2022 16:40:57 +0200 Subject: [PATCH 4/4] chore: Make linter happy --- trace/trace.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/trace/trace.go b/trace/trace.go index 1b5ffb00a1f..4aa94f79f46 100644 --- a/trace/trace.go +++ b/trace/trace.go @@ -365,8 +365,8 @@ type Span interface { // SetStatus sets the status of the Span in the form of a code and a // description, provided the status hasn't already been set to a higher - // value before (OK > Error > Unset). The description is only included in a - // status when the code is for an error. + // value before (OK > Error > Unset). The description is only included in a + // status when the code is for an error. SetStatus(code codes.Code, description string) // SetName sets the Span name.