From fb8bdc8ca6915937de5c55f832d5a0dc52445036 Mon Sep 17 00:00:00 2001 From: Steven Karis Date: Wed, 9 Oct 2019 18:15:36 -0700 Subject: [PATCH 1/5] Implement W3C Correlation Context propagator --- api/propagation/propagator.go | 22 ++ .../http_correlation_context_propagator.go | 104 ++++++++++ ...ttp_correlation_context_propagator_test.go | 191 ++++++++++++++++++ .../http_trace_context_propagator_test.go | 4 +- 4 files changed, 319 insertions(+), 2 deletions(-) create mode 100644 propagation/http_correlation_context_propagator.go create mode 100644 propagation/http_correlation_context_propagator_test.go diff --git a/api/propagation/propagator.go b/api/propagation/propagator.go index d7adb7e9fb5..52934e034f6 100644 --- a/api/propagation/propagator.go +++ b/api/propagation/propagator.go @@ -42,6 +42,28 @@ type TextFormatPropagator interface { GetAllKeys() []string } +// TextFormatBaggagePropagator is an interface that specifies methods to inject and extract +// DistributedContext into/from a carrier using Supplier interface. +// For example, HTTP Correlation Context propagator would encode DistributedContext into W3C Trace +// Correlation Context Header and set the header into HttpRequest. +type TextFormatCorrelationContextPropagator interface { + // Inject method encodes the provided key values it into propagator + // specific format and then injects the encoded distributedcontext using supplier into a carrier + // associated with the supplier. + Inject(dctx []core.KeyValue, supplier Supplier) + + // Extract method retrieves encoded DistributedContext using supplier from the associated carrier. + // It decodes the DistributedContext and returns it. If no DistributedContext was retrieved OR + // if the retrieved DistributedContext is invalid then an empty DistributedContext is returned. + Extract(ctx context.Context, supplier Supplier) []core.KeyValue + + // GetAllKeys returns all the keys that this propagator injects/extracts into/from a + // carrier. The use cases for this are + // * allow pre-allocation of fields, especially in systems like gRPC Metadata + // * allow a single-pass over an iterator (ex OpenTracing has no getter in TextMap) + GetAllKeys() []string +} + // Supplier is an interface that specifies methods to retrieve and store // value for a key to an associated carrier. // Get method retrieves the value for a given key. diff --git a/propagation/http_correlation_context_propagator.go b/propagation/http_correlation_context_propagator.go new file mode 100644 index 00000000000..429426fa51b --- /dev/null +++ b/propagation/http_correlation_context_propagator.go @@ -0,0 +1,104 @@ +// Copyright 2019, OpenTelemetry Authors +// +// 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 propagation + +import ( + "context" + "net/url" + "strings" + + "go.opentelemetry.io/api/core" + "go.opentelemetry.io/api/key" + apipropagation "go.opentelemetry.io/api/propagation" +) + +const ( + CorrelationContextHeader = "Correlation-Context" +) + +type httpCorrelationContextPropagator struct{} + +var _ apipropagation.TextFormatCorrelationContextPropagator = httpCorrelationContextPropagator{} + +func (hp httpCorrelationContextPropagator) Inject(kvs []core.KeyValue, supplier apipropagation.Supplier) { + if len(kvs) == 0 { + return + } + + var headerValueBuilder strings.Builder + firstIter := true + for _, kv := range kvs { + if !firstIter { + headerValueBuilder.WriteRune(',') + } + firstIter = false + headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Key.Name))) + headerValueBuilder.WriteRune('=') + headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Value.Emit()))) + } + supplier.Set(CorrelationContextHeader, headerValueBuilder.String()) +} + +func (hp httpCorrelationContextPropagator) Extract(ctx context.Context, supplier apipropagation.Supplier) []core.KeyValue { + correlationContext := supplier.Get(CorrelationContextHeader) + if correlationContext == "" { + return nil + } + + contextValues := strings.Split(correlationContext, ",") + keyValues := make([]core.KeyValue, 0, len(contextValues)) + for _, contextValue := range contextValues { + valueAndProps := strings.Split(contextValue, ";") + if len(valueAndProps) < 1 { + continue + } + nameValue := strings.Split(valueAndProps[0], "=") + if len(nameValue) < 2 { + continue + } + name, err := url.QueryUnescape(nameValue[0]) + if err != nil { + continue + } + trimmedName := strings.TrimSpace(name) + value, err := url.QueryUnescape(nameValue[1]) + if err != nil { + continue + } + trimmedValue := strings.TrimSpace(value) + + // TODO (skaris): properties defiend https://w3c.github.io/correlation-context/, are currently + // just put as part of the value. + var trimmedValueWithProps strings.Builder + trimmedValueWithProps.WriteString(trimmedValue) + for _, prop := range valueAndProps[1:] { + trimmedValueWithProps.WriteRune(';') + trimmedValueWithProps.WriteString(prop) + } + + keyValues = append(keyValues, key.New(trimmedName).String(trimmedValueWithProps.String())) + } + return keyValues +} + +func (hp httpCorrelationContextPropagator) GetAllKeys() []string { + return []string{CorrelationContextHeader} +} + +// HttpTraceContextPropagator creates a new text format propagator that propagates SpanContext +// in W3C TraceContext format. +func HttpCorrelationContextPropagator() apipropagation.TextFormatCorrelationContextPropagator { + return httpCorrelationContextPropagator{} +} diff --git a/propagation/http_correlation_context_propagator_test.go b/propagation/http_correlation_context_propagator_test.go new file mode 100644 index 00000000000..e6a22a2a7f9 --- /dev/null +++ b/propagation/http_correlation_context_propagator_test.go @@ -0,0 +1,191 @@ +// Copyright 2019, OpenTelemetry Authors +// +// 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 propagation_test + +import ( + "context" + "net/http" + "testing" + + "go.opentelemetry.io/api/key" + + "github.com/google/go-cmp/cmp" + + "go.opentelemetry.io/api/core" + "go.opentelemetry.io/api/trace" + mocktrace "go.opentelemetry.io/internal/trace" + "go.opentelemetry.io/propagation" +) + +func TestExtractValidDistributedContextFromHTTPReq(t *testing.T) { + trace.SetGlobalTracer(&mocktrace.MockTracer{}) + propagator := propagation.HttpCorrelationContextPropagator() + tests := []struct { + name string + header string + wantKVs []core.KeyValue + }{ + { + name: "valid w3cHeader", + header: "key1=val1,key2=val2", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid w3cHeader with spaces", + header: "key1 = val1, key2 =val2 ", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid w3cHeader with properties", + header: "key1=val1,key2=val2;prop=1", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2;prop=1"), + }, + }, + { + name: "valid header with url-escaped comma", + header: "key1=val1,key2=val2%2Cval3", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2,val3"), + }, + }, + { + name: "valid header with an invalid header", + header: "key1=val1,key2=val2,a,val3", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid header with no value", + header: "key1=,key2=val2", + wantKVs: []core.KeyValue{ + key.New("key1").String(""), + key.New("key2").String("val2"), + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + req.Header.Set("Correlation-Context", tt.header) + + ctx := context.Background() + gotSc := propagator.Extract(ctx, req.Header) + if diff := cmp.Diff(gotSc, tt.wantKVs); diff != "" { + t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) + } + }) + } +} + +func TestExtractInvalidDistributedContextFromHTTPReq(t *testing.T) { + trace.SetGlobalTracer(&mocktrace.MockTracer{}) + propagator := propagation.HttpCorrelationContextPropagator() + tests := []struct { + name string + header string + }{ + { + name: "no key values", + header: "header1", + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + req.Header.Set("Correlation-Context", tt.header) + + ctx := context.Background() + gotKvs := propagator.Extract(ctx, req.Header) + if diff := cmp.Diff(len(gotKvs), 0); diff != "" { + t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) + } + }) + } +} + +func TestInjectCorrelationContextToHTTPReq(t *testing.T) { + propagator := propagation.HttpCorrelationContextPropagator() + tests := []struct { + name string + kvs []core.KeyValue + wantHeader string + }{ + { + name: "two simple values", + kvs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + wantHeader: "key1=val1,key2=val2", + }, + { + name: "two values with escaped chars", + kvs: []core.KeyValue{ + key.New("key1").String("val1,val2"), + key.New("key2").String("val3=4"), + }, + wantHeader: "key1=val1%2Cval2,key2=val3%3D4", + }, + { + name: "values of non-string types", + kvs: []core.KeyValue{ + key.New("key1").Bool(true), + key.New("key2").Int(123), + key.New("key3").Int64(123), + key.New("key4").Int32(123), + key.New("key5").Uint(123), + key.New("key6").Uint32(123), + key.New("key7").Uint64(123), + key.New("key8").Float64(123.567), + key.New("key9").Float32(123.567), + key.New("key10").Bytes([]byte{0x68, 0x69}), + }, + wantHeader: "key1=true,key2=123,key3=123,key4=123,key5=123,key6=123,key7=123,key8=123.567,key9=123.56700134277344,key10=hi", + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + propagator.Inject(tt.kvs, req.Header) + + gotHeader := req.Header.Get("Correlation-Context") + if diff := cmp.Diff(gotHeader, tt.wantHeader); diff != "" { + t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) + } + }) + } +} + +func TestHttpCorrelationContextPropagator_GetAllKeys(t *testing.T) { + propagator := propagation.HttpCorrelationContextPropagator() + want := []string{"Correlation-Context"} + got := propagator.GetAllKeys() + if diff := cmp.Diff(got, want); diff != "" { + t.Errorf("GetAllKeys: -got +want %s", diff) + } +} diff --git a/propagation/http_trace_context_propagator_test.go b/propagation/http_trace_context_propagator_test.go index 4d7aa57577b..9e566f4c31f 100644 --- a/propagation/http_trace_context_propagator_test.go +++ b/propagation/http_trace_context_propagator_test.go @@ -42,7 +42,7 @@ func TestExtractValidTraceContextFromHTTPReq(t *testing.T) { wantSc core.SpanContext }{ { - name: "valid b3Header", + name: "valid w3cHeader", header: "00-4bf92f3577b34da6a3ce929d0e0e4736-00f067aa0ba902b7-00", wantSc: core.SpanContext{ TraceID: traceID, @@ -50,7 +50,7 @@ func TestExtractValidTraceContextFromHTTPReq(t *testing.T) { }, }, { - name: "valid b3Header and sampled", + name: "valid w3cHeader and sampled", header: "00-4bf92f3577b34da6a3ce929d0e0e4736-00f067aa0ba902b7-01", wantSc: core.SpanContext{ TraceID: traceID, From 856c789f42a56e0770d9c7910d0d5c5b06d445d2 Mon Sep 17 00:00:00 2001 From: Steven Karis Date: Wed, 16 Oct 2019 11:00:47 -0700 Subject: [PATCH 2/5] PR comments --- api/propagation/noop_propagator.go | 7 +- api/propagation/propagator.go | 15 +- plugin/httptrace/httptrace.go | 13 +- plugin/othttp/handler.go | 3 +- propagation/http_b3_propagator.go | 9 +- .../http_b3_propagator_benchmark_test.go | 5 +- propagation/http_b3_propagator_test.go | 5 +- .../http_correlation_context_propagator.go | 104 --------- ...ttp_correlation_context_propagator_test.go | 191 ---------------- propagation/http_trace_context_propagator.go | 87 +++++++- ...trace_context_propagator_benchmark_test.go | 3 +- .../http_trace_context_propagator_test.go | 205 +++++++++++++++++- 12 files changed, 318 insertions(+), 329 deletions(-) delete mode 100644 propagation/http_correlation_context_propagator.go delete mode 100644 propagation/http_correlation_context_propagator_test.go diff --git a/api/propagation/noop_propagator.go b/api/propagation/noop_propagator.go index dc185c80c92..fed50261cdf 100644 --- a/api/propagation/noop_propagator.go +++ b/api/propagation/noop_propagator.go @@ -18,6 +18,7 @@ import ( "context" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" ) // NoopTextFormatPropagator implements TextFormatPropagator that does nothing. @@ -26,12 +27,12 @@ type NoopTextFormatPropagator struct{} var _ TextFormatPropagator = NoopTextFormatPropagator{} // Inject does nothing. -func (np NoopTextFormatPropagator) Inject(ctx context.Context, supplier Supplier) { +func (np NoopTextFormatPropagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier Supplier) { } // Extract does nothing and returns an empty SpanContext -func (np NoopTextFormatPropagator) Extract(ctx context.Context, supplier Supplier) core.SpanContext { - return core.EmptySpanContext() +func (np NoopTextFormatPropagator) Extract(ctx context.Context, supplier Supplier) (core.SpanContext, dctx.Map) { + return core.EmptySpanContext(), dctx.NewEmptyMap() } // GetAllKeys returns empty list of strings. diff --git a/api/propagation/propagator.go b/api/propagation/propagator.go index 52934e034f6..b17b5bb2c42 100644 --- a/api/propagation/propagator.go +++ b/api/propagation/propagator.go @@ -18,22 +18,25 @@ import ( "context" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" ) // TextFormatPropagator is an interface that specifies methods to inject and extract SpanContext -// into/from a carrier using Supplier interface. +// and distributed context into/from a carrier using Supplier interface. // For example, HTTP Trace Context propagator would encode SpanContext into W3C Trace // Context Header and set the header into HttpRequest. type TextFormatPropagator interface { // Inject method retrieves current SpanContext from the ctx, encodes it into propagator // specific format and then injects the encoded SpanContext using supplier into a carrier - // associated with the supplier. - Inject(ctx context.Context, supplier Supplier) + // associated with the supplier. It also takes a correlationCtx whose values will be + // injected into a carrier using the supplier. + Inject(ctx context.Context, correlationCtx dctx.Map, supplier Supplier) // Extract method retrieves encoded SpanContext using supplier from the associated carrier. - // It decodes the SpanContext and returns it. If no SpanContext was retrieved OR - // if the retrieved SpanContext is invalid then an empty SpanContext is returned. - Extract(ctx context.Context, supplier Supplier) core.SpanContext + // It decodes the SpanContext and returns it and a dctx of correlated context. + // If no SpanContext was retrieved OR if the retrieved SpanContext is invalid then + // an empty SpanContext is returned. + Extract(ctx context.Context, supplier Supplier) (core.SpanContext, dctx.Map) // GetAllKeys returns all the keys that this propagator injects/extracts into/from a // carrier. The use cases for this are diff --git a/plugin/httptrace/httptrace.go b/plugin/httptrace/httptrace.go index 409fa568ee3..53b82ef6a70 100644 --- a/plugin/httptrace/httptrace.go +++ b/plugin/httptrace/httptrace.go @@ -19,6 +19,7 @@ import ( "net/http" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/key" "go.opentelemetry.io/propagation" ) @@ -36,16 +37,24 @@ var ( // Returns the Attributes, Context Entries, and SpanContext that were encoded by Inject. func Extract(ctx context.Context, req *http.Request) ([]core.KeyValue, []core.KeyValue, core.SpanContext) { - sc := propagator.Extract(ctx, req.Header) + sc, correlationCtx := propagator.Extract(ctx, req.Header) attrs := []core.KeyValue{ URLKey.String(req.URL.String()), // Etc. } + correlationCtx.Foreach(func(kv core.KeyValue) bool { + attrs = append(attrs, kv) + return true + }) return attrs, nil, sc } func Inject(ctx context.Context, req *http.Request) { - propagator.Inject(ctx, req.Header) + propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) +} + +func InjectWithCorrelationCtx(ctx context.Context, correlationCtx dctx.Map, req *http.Request) { + propagator.Inject(ctx, correlationCtx, req.Header) } diff --git a/plugin/othttp/handler.go b/plugin/othttp/handler.go index b8d0fae5cb0..c62b9b79e79 100644 --- a/plugin/othttp/handler.go +++ b/plugin/othttp/handler.go @@ -143,7 +143,8 @@ func NewHandler(handler http.Handler, operation string, opts ...Option) http.Han func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) { opts := append([]trace.SpanOption{}, h.spanOptions...) // start with the configured options - sc := h.prop.Extract(r.Context(), r.Header) + // TODO: do something with the correlation context + sc, _ := h.prop.Extract(r.Context(), r.Header) if sc.IsValid() { // not a valid span context, so no link / parent relationship to establish var opt trace.SpanOption if h.public { diff --git a/propagation/http_b3_propagator.go b/propagation/http_b3_propagator.go index e378bed70e0..eb23c6fa4ab 100644 --- a/propagation/http_b3_propagator.go +++ b/propagation/http_b3_propagator.go @@ -24,6 +24,7 @@ import ( "go.opentelemetry.io/api/trace" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" apipropagation "go.opentelemetry.io/api/propagation" ) @@ -59,7 +60,7 @@ var _ apipropagation.TextFormatPropagator = HTTPB3Propagator{} var hexStr32ByteRegex = regexp.MustCompile("^[a-f0-9]{32}$") var hexStr16ByteRegex = regexp.MustCompile("^[a-f0-9]{16}$") -func (b3 HTTPB3Propagator) Inject(ctx context.Context, supplier apipropagation.Supplier) { +func (b3 HTTPB3Propagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier apipropagation.Supplier) { sc := trace.CurrentSpan(ctx).SpanContext() if sc.IsValid() { if b3.SingleHeader { @@ -84,11 +85,11 @@ func (b3 HTTPB3Propagator) Inject(ctx context.Context, supplier apipropagation.S } // Extract retrieves B3 Headers from the supplier -func (b3 HTTPB3Propagator) Extract(ctx context.Context, supplier apipropagation.Supplier) core.SpanContext { +func (b3 HTTPB3Propagator) Extract(ctx context.Context, supplier apipropagation.Supplier) (core.SpanContext, dctx.Map) { if b3.SingleHeader { - return b3.extractSingleHeader(supplier) + return b3.extractSingleHeader(supplier), dctx.NewEmptyMap() } - return b3.extract(supplier) + return b3.extract(supplier), dctx.NewEmptyMap() } func (b3 HTTPB3Propagator) GetAllKeys() []string { diff --git a/propagation/http_b3_propagator_benchmark_test.go b/propagation/http_b3_propagator_benchmark_test.go index beed63df3d3..6dc67965744 100644 --- a/propagation/http_b3_propagator_benchmark_test.go +++ b/propagation/http_b3_propagator_benchmark_test.go @@ -19,6 +19,7 @@ import ( "net/http" "testing" + dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" "go.opentelemetry.io/propagation" @@ -65,7 +66,7 @@ func BenchmarkExtractB3(b *testing.B) { b.ReportAllocs() b.ResetTimer() for i := 0; i < b.N; i++ { - _ = propagator.Extract(ctx, req.Header) + _, _ = propagator.Extract(ctx, req.Header) } }) } @@ -112,7 +113,7 @@ func BenchmarkInjectB3(b *testing.B) { b.ReportAllocs() b.ResetTimer() for i := 0; i < b.N; i++ { - propagator.Inject(ctx, req.Header) + propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) } }) } diff --git a/propagation/http_b3_propagator_test.go b/propagation/http_b3_propagator_test.go index 68baa8fc592..e23b2d81380 100644 --- a/propagation/http_b3_propagator_test.go +++ b/propagation/http_b3_propagator_test.go @@ -21,6 +21,7 @@ import ( "github.com/google/go-cmp/cmp" + dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" "go.opentelemetry.io/propagation" @@ -65,7 +66,7 @@ func TestExtractB3(t *testing.T) { } ctx := context.Background() - gotSc := propagator.Extract(ctx, req.Header) + gotSc, _ := propagator.Extract(ctx, req.Header) if diff := cmp.Diff(gotSc, tt.wantSc); diff != "" { t.Errorf("%s: %s: -got +want %s", tg.name, tt.name, diff) } @@ -111,7 +112,7 @@ func TestInjectB3(t *testing.T) { } else { ctx, _ = mockTracer.Start(ctx, "inject") } - propagator.Inject(ctx, req.Header) + propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) for h, v := range tt.wantHeaders { got, want := req.Header.Get(h), v diff --git a/propagation/http_correlation_context_propagator.go b/propagation/http_correlation_context_propagator.go deleted file mode 100644 index 429426fa51b..00000000000 --- a/propagation/http_correlation_context_propagator.go +++ /dev/null @@ -1,104 +0,0 @@ -// Copyright 2019, OpenTelemetry Authors -// -// 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 propagation - -import ( - "context" - "net/url" - "strings" - - "go.opentelemetry.io/api/core" - "go.opentelemetry.io/api/key" - apipropagation "go.opentelemetry.io/api/propagation" -) - -const ( - CorrelationContextHeader = "Correlation-Context" -) - -type httpCorrelationContextPropagator struct{} - -var _ apipropagation.TextFormatCorrelationContextPropagator = httpCorrelationContextPropagator{} - -func (hp httpCorrelationContextPropagator) Inject(kvs []core.KeyValue, supplier apipropagation.Supplier) { - if len(kvs) == 0 { - return - } - - var headerValueBuilder strings.Builder - firstIter := true - for _, kv := range kvs { - if !firstIter { - headerValueBuilder.WriteRune(',') - } - firstIter = false - headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Key.Name))) - headerValueBuilder.WriteRune('=') - headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Value.Emit()))) - } - supplier.Set(CorrelationContextHeader, headerValueBuilder.String()) -} - -func (hp httpCorrelationContextPropagator) Extract(ctx context.Context, supplier apipropagation.Supplier) []core.KeyValue { - correlationContext := supplier.Get(CorrelationContextHeader) - if correlationContext == "" { - return nil - } - - contextValues := strings.Split(correlationContext, ",") - keyValues := make([]core.KeyValue, 0, len(contextValues)) - for _, contextValue := range contextValues { - valueAndProps := strings.Split(contextValue, ";") - if len(valueAndProps) < 1 { - continue - } - nameValue := strings.Split(valueAndProps[0], "=") - if len(nameValue) < 2 { - continue - } - name, err := url.QueryUnescape(nameValue[0]) - if err != nil { - continue - } - trimmedName := strings.TrimSpace(name) - value, err := url.QueryUnescape(nameValue[1]) - if err != nil { - continue - } - trimmedValue := strings.TrimSpace(value) - - // TODO (skaris): properties defiend https://w3c.github.io/correlation-context/, are currently - // just put as part of the value. - var trimmedValueWithProps strings.Builder - trimmedValueWithProps.WriteString(trimmedValue) - for _, prop := range valueAndProps[1:] { - trimmedValueWithProps.WriteRune(';') - trimmedValueWithProps.WriteString(prop) - } - - keyValues = append(keyValues, key.New(trimmedName).String(trimmedValueWithProps.String())) - } - return keyValues -} - -func (hp httpCorrelationContextPropagator) GetAllKeys() []string { - return []string{CorrelationContextHeader} -} - -// HttpTraceContextPropagator creates a new text format propagator that propagates SpanContext -// in W3C TraceContext format. -func HttpCorrelationContextPropagator() apipropagation.TextFormatCorrelationContextPropagator { - return httpCorrelationContextPropagator{} -} diff --git a/propagation/http_correlation_context_propagator_test.go b/propagation/http_correlation_context_propagator_test.go deleted file mode 100644 index e6a22a2a7f9..00000000000 --- a/propagation/http_correlation_context_propagator_test.go +++ /dev/null @@ -1,191 +0,0 @@ -// Copyright 2019, OpenTelemetry Authors -// -// 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 propagation_test - -import ( - "context" - "net/http" - "testing" - - "go.opentelemetry.io/api/key" - - "github.com/google/go-cmp/cmp" - - "go.opentelemetry.io/api/core" - "go.opentelemetry.io/api/trace" - mocktrace "go.opentelemetry.io/internal/trace" - "go.opentelemetry.io/propagation" -) - -func TestExtractValidDistributedContextFromHTTPReq(t *testing.T) { - trace.SetGlobalTracer(&mocktrace.MockTracer{}) - propagator := propagation.HttpCorrelationContextPropagator() - tests := []struct { - name string - header string - wantKVs []core.KeyValue - }{ - { - name: "valid w3cHeader", - header: "key1=val1,key2=val2", - wantKVs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2"), - }, - }, - { - name: "valid w3cHeader with spaces", - header: "key1 = val1, key2 =val2 ", - wantKVs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2"), - }, - }, - { - name: "valid w3cHeader with properties", - header: "key1=val1,key2=val2;prop=1", - wantKVs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2;prop=1"), - }, - }, - { - name: "valid header with url-escaped comma", - header: "key1=val1,key2=val2%2Cval3", - wantKVs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2,val3"), - }, - }, - { - name: "valid header with an invalid header", - header: "key1=val1,key2=val2,a,val3", - wantKVs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2"), - }, - }, - { - name: "valid header with no value", - header: "key1=,key2=val2", - wantKVs: []core.KeyValue{ - key.New("key1").String(""), - key.New("key2").String("val2"), - }, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - req, _ := http.NewRequest("GET", "http://example.com", nil) - req.Header.Set("Correlation-Context", tt.header) - - ctx := context.Background() - gotSc := propagator.Extract(ctx, req.Header) - if diff := cmp.Diff(gotSc, tt.wantKVs); diff != "" { - t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) - } - }) - } -} - -func TestExtractInvalidDistributedContextFromHTTPReq(t *testing.T) { - trace.SetGlobalTracer(&mocktrace.MockTracer{}) - propagator := propagation.HttpCorrelationContextPropagator() - tests := []struct { - name string - header string - }{ - { - name: "no key values", - header: "header1", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - req, _ := http.NewRequest("GET", "http://example.com", nil) - req.Header.Set("Correlation-Context", tt.header) - - ctx := context.Background() - gotKvs := propagator.Extract(ctx, req.Header) - if diff := cmp.Diff(len(gotKvs), 0); diff != "" { - t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) - } - }) - } -} - -func TestInjectCorrelationContextToHTTPReq(t *testing.T) { - propagator := propagation.HttpCorrelationContextPropagator() - tests := []struct { - name string - kvs []core.KeyValue - wantHeader string - }{ - { - name: "two simple values", - kvs: []core.KeyValue{ - key.New("key1").String("val1"), - key.New("key2").String("val2"), - }, - wantHeader: "key1=val1,key2=val2", - }, - { - name: "two values with escaped chars", - kvs: []core.KeyValue{ - key.New("key1").String("val1,val2"), - key.New("key2").String("val3=4"), - }, - wantHeader: "key1=val1%2Cval2,key2=val3%3D4", - }, - { - name: "values of non-string types", - kvs: []core.KeyValue{ - key.New("key1").Bool(true), - key.New("key2").Int(123), - key.New("key3").Int64(123), - key.New("key4").Int32(123), - key.New("key5").Uint(123), - key.New("key6").Uint32(123), - key.New("key7").Uint64(123), - key.New("key8").Float64(123.567), - key.New("key9").Float32(123.567), - key.New("key10").Bytes([]byte{0x68, 0x69}), - }, - wantHeader: "key1=true,key2=123,key3=123,key4=123,key5=123,key6=123,key7=123,key8=123.567,key9=123.56700134277344,key10=hi", - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - req, _ := http.NewRequest("GET", "http://example.com", nil) - propagator.Inject(tt.kvs, req.Header) - - gotHeader := req.Header.Get("Correlation-Context") - if diff := cmp.Diff(gotHeader, tt.wantHeader); diff != "" { - t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) - } - }) - } -} - -func TestHttpCorrelationContextPropagator_GetAllKeys(t *testing.T) { - propagator := propagation.HttpCorrelationContextPropagator() - want := []string{"Correlation-Context"} - got := propagator.GetAllKeys() - if diff := cmp.Diff(got, want); diff != "" { - t.Errorf("GetAllKeys: -got +want %s", diff) - } -} diff --git a/propagation/http_trace_context_propagator.go b/propagation/http_trace_context_propagator.go index f1ebfce4f4c..719dc7570c3 100644 --- a/propagation/http_trace_context_propagator.go +++ b/propagation/http_trace_context_propagator.go @@ -18,20 +18,24 @@ import ( "context" "encoding/hex" "fmt" + "net/url" "regexp" "strconv" "strings" - "go.opentelemetry.io/api/trace" + "go.opentelemetry.io/api/key" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" apipropagation "go.opentelemetry.io/api/propagation" + "go.opentelemetry.io/api/trace" ) const ( - supportedVersion = 0 - maxVersion = 254 - TraceparentHeader = "Traceparent" + supportedVersion = 0 + maxVersion = 254 + TraceparentHeader = "Traceparent" + CorrelationContextHeader = "Correlation-Context" ) // HTTPTraceContextPropagator propagates SpanContext in W3C TraceContext format. @@ -40,7 +44,7 @@ type HTTPTraceContextPropagator struct{} var _ apipropagation.TextFormatPropagator = HTTPTraceContextPropagator{} var traceCtxRegExp = regexp.MustCompile("^[0-9a-f]{2}-[a-f0-9]{32}-[a-f0-9]{16}-[a-f0-9]{2}-?") -func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, supplier apipropagation.Supplier) { +func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier apipropagation.Supplier) { sc := trace.CurrentSpan(ctx).SpanContext() if sc.IsValid() { h := fmt.Sprintf("%.2x-%.16x%.16x-%.16x-%.2x", @@ -51,9 +55,34 @@ func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, supplier apipro sc.TraceFlags&core.TraceFlagsSampled) supplier.Set(TraceparentHeader, h) } + + firstIter := true + var headerValueBuilder strings.Builder + correlationCtx.Foreach(func(kv core.KeyValue) bool { + if !firstIter { + headerValueBuilder.WriteRune(',') + } + firstIter = false + headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Key.Name))) + headerValueBuilder.WriteRune('=') + headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Value.Emit()))) + return true + }) + if headerValueBuilder.Len() > 0 { + headerString := headerValueBuilder.String() + supplier.Set(CorrelationContextHeader, headerString) + } } -func (hp HTTPTraceContextPropagator) Extract(ctx context.Context, supplier apipropagation.Supplier) core.SpanContext { +func (hp HTTPTraceContextPropagator) Extract( + ctx context.Context, supplier apipropagation.Supplier, +) (core.SpanContext, dctx.Map) { + return hp.extractSpanContext(ctx, supplier), hp.extractCorrelationCtx(ctx, supplier) +} + +func (hp HTTPTraceContextPropagator) extractSpanContext( + ctx context.Context, supplier apipropagation.Supplier, +) core.SpanContext { h := supplier.Get(TraceparentHeader) if h == "" { return core.EmptySpanContext() @@ -128,6 +157,50 @@ func (hp HTTPTraceContextPropagator) Extract(ctx context.Context, supplier apipr return sc } +func (hp HTTPTraceContextPropagator) extractCorrelationCtx(ctx context.Context, supplier apipropagation.Supplier) dctx.Map { + correlationContext := supplier.Get(CorrelationContextHeader) + if correlationContext == "" { + return dctx.NewEmptyMap() + } + + contextValues := strings.Split(correlationContext, ",") + keyValues := make([]core.KeyValue, 0, len(contextValues)) + for _, contextValue := range contextValues { + valueAndProps := strings.Split(contextValue, ";") + if len(valueAndProps) < 1 { + continue + } + nameValue := strings.Split(valueAndProps[0], "=") + if len(nameValue) < 2 { + continue + } + name, err := url.QueryUnescape(nameValue[0]) + if err != nil { + continue + } + trimmedName := strings.TrimSpace(name) + value, err := url.QueryUnescape(nameValue[1]) + if err != nil { + continue + } + trimmedValue := strings.TrimSpace(value) + + // TODO (skaris): properties defiend https://w3c.github.io/correlation-context/, are currently + // just put as part of the value. + var trimmedValueWithProps strings.Builder + trimmedValueWithProps.WriteString(trimmedValue) + for _, prop := range valueAndProps[1:] { + trimmedValueWithProps.WriteRune(';') + trimmedValueWithProps.WriteString(prop) + } + + keyValues = append(keyValues, key.New(trimmedName).String(trimmedValueWithProps.String())) + } + return dctx.NewMap(dctx.MapUpdate{ + MultiKV: keyValues, + }) +} + func (hp HTTPTraceContextPropagator) GetAllKeys() []string { - return []string{TraceparentHeader} + return []string{TraceparentHeader, CorrelationContextHeader} } diff --git a/propagation/http_trace_context_propagator_benchmark_test.go b/propagation/http_trace_context_propagator_benchmark_test.go index 660b4d1609f..2447610ff5d 100644 --- a/propagation/http_trace_context_propagator_benchmark_test.go +++ b/propagation/http_trace_context_propagator_benchmark_test.go @@ -6,6 +6,7 @@ import ( "testing" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" ) @@ -17,7 +18,7 @@ func BenchmarkInject(b *testing.B) { req, _ := http.NewRequest("GET", "http://example.com", nil) b.ResetTimer() for i := 0; i < b.N; i++ { - t.Inject(ctx, req.Header) + t.Inject(ctx, dctx.NewEmptyMap(), req.Header) } }) } diff --git a/propagation/http_trace_context_propagator_test.go b/propagation/http_trace_context_propagator_test.go index 9e566f4c31f..cac9d657d52 100644 --- a/propagation/http_trace_context_propagator_test.go +++ b/propagation/http_trace_context_propagator_test.go @@ -17,13 +17,15 @@ package propagation_test import ( "context" "net/http" + "strings" "testing" - "go.opentelemetry.io/api/trace" - "github.com/google/go-cmp/cmp" "go.opentelemetry.io/api/core" + dctx "go.opentelemetry.io/api/distributedcontext" + "go.opentelemetry.io/api/key" + "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" "go.opentelemetry.io/propagation" ) @@ -119,7 +121,7 @@ func TestExtractValidTraceContextFromHTTPReq(t *testing.T) { req.Header.Set("traceparent", tt.header) ctx := context.Background() - gotSc := propagator.Extract(ctx, req.Header) + gotSc, _ := propagator.Extract(ctx, req.Header) if diff := cmp.Diff(gotSc, tt.wantSc); diff != "" { t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) } @@ -207,7 +209,7 @@ func TestExtractInvalidTraceContextFromHTTPReq(t *testing.T) { req.Header.Set("traceparent", tt.header) ctx := context.Background() - gotSc := propagator.Extract(ctx, req.Header) + gotSc, _ := propagator.Extract(ctx, req.Header) if diff := cmp.Diff(gotSc, wantSc); diff != "" { t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, diff) } @@ -266,7 +268,7 @@ func TestInjectTraceContextToHTTPReq(t *testing.T) { if tt.sc.IsValid() { ctx, _ = mockTracer.Start(ctx, "inject", trace.ChildOf(tt.sc)) } - propagator.Inject(ctx, req.Header) + propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) gotHeader := req.Header.Get("traceparent") if diff := cmp.Diff(gotHeader, tt.wantHeader); diff != "" { @@ -276,9 +278,200 @@ func TestInjectTraceContextToHTTPReq(t *testing.T) { } } +func TestExtractValidDistributedContextFromHTTPReq(t *testing.T) { + trace.SetGlobalTracer(&mocktrace.MockTracer{}) + propagator := propagation.HTTPTraceContextPropagator{} + tests := []struct { + name string + header string + wantKVs []core.KeyValue + }{ + { + name: "valid w3cHeader", + header: "key1=val1,key2=val2", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid w3cHeader with spaces", + header: "key1 = val1, key2 =val2 ", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid w3cHeader with properties", + header: "key1=val1,key2=val2;prop=1", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2;prop=1"), + }, + }, + { + name: "valid header with url-escaped comma", + header: "key1=val1,key2=val2%2Cval3", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2,val3"), + }, + }, + { + name: "valid header with an invalid header", + header: "key1=val1,key2=val2,a,val3", + wantKVs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + }, + { + name: "valid header with no value", + header: "key1=,key2=val2", + wantKVs: []core.KeyValue{ + key.New("key1").String(""), + key.New("key2").String("val2"), + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + req.Header.Set("Correlation-Context", tt.header) + + ctx := context.Background() + _, gotCorCtx := propagator.Extract(ctx, req.Header) + wantCorCtx := dctx.NewMap(dctx.MapUpdate{MultiKV: tt.wantKVs}) + if gotCorCtx.Len() != wantCorCtx.Len() { + t.Errorf( + "Got and Want CorCtx are not the same size %d != %d", + gotCorCtx.Len(), + wantCorCtx.Len(), + ) + } + totalDiff := "" + wantCorCtx.Foreach(func(kv core.KeyValue) bool { + val, _ := gotCorCtx.Value(kv.Key) + diff := cmp.Diff(kv, core.KeyValue{Key: kv.Key, Value: val}) + if diff != "" { + totalDiff += diff + "\n" + } + return true + }) + if totalDiff != "" { + t.Errorf("Extract Tracecontext: %s: -got +want %s", tt.name, totalDiff) + } + }) + } +} + +func TestExtractInvalidDistributedContextFromHTTPReq(t *testing.T) { + trace.SetGlobalTracer(&mocktrace.MockTracer{}) + propagator := propagation.HTTPTraceContextPropagator{} + tests := []struct { + name string + header string + }{ + { + name: "no key values", + header: "header1", + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + req.Header.Set("Correlation-Context", tt.header) + + ctx := context.Background() + _, gotCorCtx := propagator.Extract(ctx, req.Header) + if gotCorCtx.Len() != 0 { + t.Errorf("Got and Want CorCtx are not the same size %d != %d", gotCorCtx.Len(), 0) + } + }) + } +} + +func TestInjectCorrelationContextToHTTPReq(t *testing.T) { + propagator := propagation.HTTPTraceContextPropagator{} + tests := []struct { + name string + kvs []core.KeyValue + wantInHeader []string + wantedLen int + }{ + { + name: "two simple values", + kvs: []core.KeyValue{ + key.New("key1").String("val1"), + key.New("key2").String("val2"), + }, + wantInHeader: []string{"key1=val1", "key2=val2"}, + }, + { + name: "two values with escaped chars", + kvs: []core.KeyValue{ + key.New("key1").String("val1,val2"), + key.New("key2").String("val3=4"), + }, + wantInHeader: []string{"key1=val1%2Cval2", "key2=val3%3D4"}, + }, + { + name: "values of non-string types", + kvs: []core.KeyValue{ + key.New("key1").Bool(true), + key.New("key2").Int(123), + key.New("key3").Int64(123), + key.New("key4").Int32(123), + key.New("key5").Uint(123), + key.New("key6").Uint32(123), + key.New("key7").Uint64(123), + key.New("key8").Float64(123.567), + key.New("key9").Float32(123.567), + key.New("key10").Bytes([]byte{0x68, 0x69}), + }, + wantInHeader: []string{ + "key1=true", + "key2=123", + "key3=123", + "key4=123", + "key5=123", + "key6=123", + "key7=123", + "key8=123.567", + "key9=123.56700134277344", + "key10=hi", + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + req, _ := http.NewRequest("GET", "http://example.com", nil) + propagator.Inject(context.Background(), dctx.NewMap(dctx.MapUpdate{MultiKV: tt.kvs}), req.Header) + + gotHeader := req.Header.Get("Correlation-Context") + wantedLen := len(strings.Join(tt.wantInHeader, ",")) + if wantedLen != len(gotHeader) { + t.Errorf( + "%s: Inject Correlation-Context incorrect length %d != %d.", tt.name, tt.wantedLen, len(gotHeader), + ) + } + for _, inHeader := range tt.wantInHeader { + if !strings.Contains(gotHeader, inHeader) { + t.Errorf( + "%s: Inject Correlation-Context missing part of header: %s in %s", tt.name, inHeader, gotHeader, + ) + } + } + }) + } +} + func TestHttpTraceContextPropagator_GetAllKeys(t *testing.T) { var propagator propagation.HTTPTraceContextPropagator - want := []string{"Traceparent"} + want := []string{"Traceparent", "Correlation-Context"} got := propagator.GetAllKeys() if diff := cmp.Diff(got, want); diff != "" { t.Errorf("GetAllKeys: -got +want %s", diff) From 788fe97520929888bc1720c819743b0bcfdfcebd Mon Sep 17 00:00:00 2001 From: Steven Karis Date: Wed, 16 Oct 2019 14:56:43 -0700 Subject: [PATCH 3/5] PR comments --- api/propagation/noop_propagator.go | 2 +- api/propagation/propagator.go | 24 +------------------ plugin/httptrace/httptrace.go | 13 ++++------ propagation/http_b3_propagator.go | 2 +- .../http_b3_propagator_benchmark_test.go | 3 +-- propagation/http_b3_propagator_test.go | 3 +-- propagation/http_trace_context_propagator.go | 6 ++--- ...trace_context_propagator_benchmark_test.go | 3 +-- .../http_trace_context_propagator_test.go | 5 ++-- 9 files changed, 17 insertions(+), 44 deletions(-) diff --git a/api/propagation/noop_propagator.go b/api/propagation/noop_propagator.go index fed50261cdf..9e74479216d 100644 --- a/api/propagation/noop_propagator.go +++ b/api/propagation/noop_propagator.go @@ -27,7 +27,7 @@ type NoopTextFormatPropagator struct{} var _ TextFormatPropagator = NoopTextFormatPropagator{} // Inject does nothing. -func (np NoopTextFormatPropagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier Supplier) { +func (np NoopTextFormatPropagator) Inject(ctx context.Context, supplier Supplier) { } // Extract does nothing and returns an empty SpanContext diff --git a/api/propagation/propagator.go b/api/propagation/propagator.go index b17b5bb2c42..3da0d287a76 100644 --- a/api/propagation/propagator.go +++ b/api/propagation/propagator.go @@ -30,7 +30,7 @@ type TextFormatPropagator interface { // specific format and then injects the encoded SpanContext using supplier into a carrier // associated with the supplier. It also takes a correlationCtx whose values will be // injected into a carrier using the supplier. - Inject(ctx context.Context, correlationCtx dctx.Map, supplier Supplier) + Inject(ctx context.Context, supplier Supplier) // Extract method retrieves encoded SpanContext using supplier from the associated carrier. // It decodes the SpanContext and returns it and a dctx of correlated context. @@ -45,28 +45,6 @@ type TextFormatPropagator interface { GetAllKeys() []string } -// TextFormatBaggagePropagator is an interface that specifies methods to inject and extract -// DistributedContext into/from a carrier using Supplier interface. -// For example, HTTP Correlation Context propagator would encode DistributedContext into W3C Trace -// Correlation Context Header and set the header into HttpRequest. -type TextFormatCorrelationContextPropagator interface { - // Inject method encodes the provided key values it into propagator - // specific format and then injects the encoded distributedcontext using supplier into a carrier - // associated with the supplier. - Inject(dctx []core.KeyValue, supplier Supplier) - - // Extract method retrieves encoded DistributedContext using supplier from the associated carrier. - // It decodes the DistributedContext and returns it. If no DistributedContext was retrieved OR - // if the retrieved DistributedContext is invalid then an empty DistributedContext is returned. - Extract(ctx context.Context, supplier Supplier) []core.KeyValue - - // GetAllKeys returns all the keys that this propagator injects/extracts into/from a - // carrier. The use cases for this are - // * allow pre-allocation of fields, especially in systems like gRPC Metadata - // * allow a single-pass over an iterator (ex OpenTracing has no getter in TextMap) - GetAllKeys() []string -} - // Supplier is an interface that specifies methods to retrieve and store // value for a key to an associated carrier. // Get method retrieves the value for a given key. diff --git a/plugin/httptrace/httptrace.go b/plugin/httptrace/httptrace.go index 53b82ef6a70..b6efcfd9d9f 100644 --- a/plugin/httptrace/httptrace.go +++ b/plugin/httptrace/httptrace.go @@ -19,7 +19,6 @@ import ( "net/http" "go.opentelemetry.io/api/core" - dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/key" "go.opentelemetry.io/propagation" ) @@ -43,18 +42,16 @@ func Extract(ctx context.Context, req *http.Request) ([]core.KeyValue, []core.Ke URLKey.String(req.URL.String()), // Etc. } + + var correlationCtxKVs []core.KeyValue correlationCtx.Foreach(func(kv core.KeyValue) bool { - attrs = append(attrs, kv) + correlationCtxKVs = append(correlationCtxKVs, kv) return true }) - return attrs, nil, sc + return attrs, correlationCtxKVs, sc } func Inject(ctx context.Context, req *http.Request) { - propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) -} - -func InjectWithCorrelationCtx(ctx context.Context, correlationCtx dctx.Map, req *http.Request) { - propagator.Inject(ctx, correlationCtx, req.Header) + propagator.Inject(ctx, req.Header) } diff --git a/propagation/http_b3_propagator.go b/propagation/http_b3_propagator.go index eb23c6fa4ab..90fca860542 100644 --- a/propagation/http_b3_propagator.go +++ b/propagation/http_b3_propagator.go @@ -60,7 +60,7 @@ var _ apipropagation.TextFormatPropagator = HTTPB3Propagator{} var hexStr32ByteRegex = regexp.MustCompile("^[a-f0-9]{32}$") var hexStr16ByteRegex = regexp.MustCompile("^[a-f0-9]{16}$") -func (b3 HTTPB3Propagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier apipropagation.Supplier) { +func (b3 HTTPB3Propagator) Inject(ctx context.Context, supplier apipropagation.Supplier) { sc := trace.CurrentSpan(ctx).SpanContext() if sc.IsValid() { if b3.SingleHeader { diff --git a/propagation/http_b3_propagator_benchmark_test.go b/propagation/http_b3_propagator_benchmark_test.go index 6dc67965744..52169b63c3e 100644 --- a/propagation/http_b3_propagator_benchmark_test.go +++ b/propagation/http_b3_propagator_benchmark_test.go @@ -19,7 +19,6 @@ import ( "net/http" "testing" - dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" "go.opentelemetry.io/propagation" @@ -113,7 +112,7 @@ func BenchmarkInjectB3(b *testing.B) { b.ReportAllocs() b.ResetTimer() for i := 0; i < b.N; i++ { - propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) + propagator.Inject(ctx, req.Header) } }) } diff --git a/propagation/http_b3_propagator_test.go b/propagation/http_b3_propagator_test.go index e23b2d81380..8e0e386d307 100644 --- a/propagation/http_b3_propagator_test.go +++ b/propagation/http_b3_propagator_test.go @@ -21,7 +21,6 @@ import ( "github.com/google/go-cmp/cmp" - dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" "go.opentelemetry.io/propagation" @@ -112,7 +111,7 @@ func TestInjectB3(t *testing.T) { } else { ctx, _ = mockTracer.Start(ctx, "inject") } - propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) + propagator.Inject(ctx, req.Header) for h, v := range tt.wantHeaders { got, want := req.Header.Get(h), v diff --git a/propagation/http_trace_context_propagator.go b/propagation/http_trace_context_propagator.go index 719dc7570c3..a5f98fe1ff8 100644 --- a/propagation/http_trace_context_propagator.go +++ b/propagation/http_trace_context_propagator.go @@ -23,10 +23,9 @@ import ( "strconv" "strings" - "go.opentelemetry.io/api/key" - "go.opentelemetry.io/api/core" dctx "go.opentelemetry.io/api/distributedcontext" + "go.opentelemetry.io/api/key" apipropagation "go.opentelemetry.io/api/propagation" "go.opentelemetry.io/api/trace" ) @@ -44,7 +43,7 @@ type HTTPTraceContextPropagator struct{} var _ apipropagation.TextFormatPropagator = HTTPTraceContextPropagator{} var traceCtxRegExp = regexp.MustCompile("^[0-9a-f]{2}-[a-f0-9]{32}-[a-f0-9]{16}-[a-f0-9]{2}-?") -func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, correlationCtx dctx.Map, supplier apipropagation.Supplier) { +func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, supplier apipropagation.Supplier) { sc := trace.CurrentSpan(ctx).SpanContext() if sc.IsValid() { h := fmt.Sprintf("%.2x-%.16x%.16x-%.16x-%.2x", @@ -56,6 +55,7 @@ func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, correlationCtx supplier.Set(TraceparentHeader, h) } + correlationCtx := dctx.FromContext(ctx) firstIter := true var headerValueBuilder strings.Builder correlationCtx.Foreach(func(kv core.KeyValue) bool { diff --git a/propagation/http_trace_context_propagator_benchmark_test.go b/propagation/http_trace_context_propagator_benchmark_test.go index 2447610ff5d..660b4d1609f 100644 --- a/propagation/http_trace_context_propagator_benchmark_test.go +++ b/propagation/http_trace_context_propagator_benchmark_test.go @@ -6,7 +6,6 @@ import ( "testing" "go.opentelemetry.io/api/core" - dctx "go.opentelemetry.io/api/distributedcontext" "go.opentelemetry.io/api/trace" mocktrace "go.opentelemetry.io/internal/trace" ) @@ -18,7 +17,7 @@ func BenchmarkInject(b *testing.B) { req, _ := http.NewRequest("GET", "http://example.com", nil) b.ResetTimer() for i := 0; i < b.N; i++ { - t.Inject(ctx, dctx.NewEmptyMap(), req.Header) + t.Inject(ctx, req.Header) } }) } diff --git a/propagation/http_trace_context_propagator_test.go b/propagation/http_trace_context_propagator_test.go index cac9d657d52..c0706545c76 100644 --- a/propagation/http_trace_context_propagator_test.go +++ b/propagation/http_trace_context_propagator_test.go @@ -268,7 +268,7 @@ func TestInjectTraceContextToHTTPReq(t *testing.T) { if tt.sc.IsValid() { ctx, _ = mockTracer.Start(ctx, "inject", trace.ChildOf(tt.sc)) } - propagator.Inject(ctx, dctx.NewEmptyMap(), req.Header) + propagator.Inject(ctx, req.Header) gotHeader := req.Header.Get("traceparent") if diff := cmp.Diff(gotHeader, tt.wantHeader); diff != "" { @@ -449,7 +449,8 @@ func TestInjectCorrelationContextToHTTPReq(t *testing.T) { for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { req, _ := http.NewRequest("GET", "http://example.com", nil) - propagator.Inject(context.Background(), dctx.NewMap(dctx.MapUpdate{MultiKV: tt.kvs}), req.Header) + ctx := dctx.NewContext(context.Background()) + propagator.Inject(ctx, req.Header) gotHeader := req.Header.Get("Correlation-Context") wantedLen := len(strings.Join(tt.wantInHeader, ",")) From baaa8ca161a75f584a3c54bb4e76788e954bc394 Mon Sep 17 00:00:00 2001 From: Steven Karis Date: Wed, 16 Oct 2019 15:26:01 -0700 Subject: [PATCH 4/5] Update test to inject context properly --- propagation/http_trace_context_propagator_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/propagation/http_trace_context_propagator_test.go b/propagation/http_trace_context_propagator_test.go index c0706545c76..536a9f843a1 100644 --- a/propagation/http_trace_context_propagator_test.go +++ b/propagation/http_trace_context_propagator_test.go @@ -449,7 +449,7 @@ func TestInjectCorrelationContextToHTTPReq(t *testing.T) { for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { req, _ := http.NewRequest("GET", "http://example.com", nil) - ctx := dctx.NewContext(context.Background()) + ctx := dctx.WithMap(context.Background(), dctx.NewMap(dctx.MapUpdate{MultiKV: tt.kvs})) propagator.Inject(ctx, req.Header) gotHeader := req.Header.Get("Correlation-Context") From c138077ce3db5cc1c43d7230b7626829cd6d063b Mon Sep 17 00:00:00 2001 From: Steven Karis Date: Thu, 17 Oct 2019 10:16:48 -0700 Subject: [PATCH 5/5] Fix merge --- propagation/http_trace_context_propagator.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/propagation/http_trace_context_propagator.go b/propagation/http_trace_context_propagator.go index a5f98fe1ff8..1cf79292fda 100644 --- a/propagation/http_trace_context_propagator.go +++ b/propagation/http_trace_context_propagator.go @@ -63,7 +63,7 @@ func (hp HTTPTraceContextPropagator) Inject(ctx context.Context, supplier apipro headerValueBuilder.WriteRune(',') } firstIter = false - headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Key.Name))) + headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace((string)(kv.Key)))) headerValueBuilder.WriteRune('=') headerValueBuilder.WriteString(url.QueryEscape(strings.TrimSpace(kv.Value.Emit()))) return true