124 lines
3.6 KiB
Go
124 lines
3.6 KiB
Go
|
// Copyright 2018, OpenCensus 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 b3 contains a propagation.HTTPFormat implementation
|
||
|
// for B3 propagation. See https://github.com/openzipkin/b3-propagation
|
||
|
// for more details.
|
||
|
package b3 // import "go.opencensus.io/plugin/ochttp/propagation/b3"
|
||
|
|
||
|
import (
|
||
|
"encoding/hex"
|
||
|
"net/http"
|
||
|
|
||
|
"go.opencensus.io/trace"
|
||
|
"go.opencensus.io/trace/propagation"
|
||
|
)
|
||
|
|
||
|
// B3 headers that OpenCensus understands.
|
||
|
const (
|
||
|
TraceIDHeader = "X-B3-TraceId"
|
||
|
SpanIDHeader = "X-B3-SpanId"
|
||
|
SampledHeader = "X-B3-Sampled"
|
||
|
)
|
||
|
|
||
|
// HTTPFormat implements propagation.HTTPFormat to propagate
|
||
|
// traces in HTTP headers in B3 propagation format.
|
||
|
// HTTPFormat skips the X-B3-ParentId and X-B3-Flags headers
|
||
|
// because there are additional fields not represented in the
|
||
|
// OpenCensus span context. Spans created from the incoming
|
||
|
// header will be the direct children of the client-side span.
|
||
|
// Similarly, reciever of the outgoing spans should use client-side
|
||
|
// span created by OpenCensus as the parent.
|
||
|
type HTTPFormat struct{}
|
||
|
|
||
|
var _ propagation.HTTPFormat = (*HTTPFormat)(nil)
|
||
|
|
||
|
// SpanContextFromRequest extracts a B3 span context from incoming requests.
|
||
|
func (f *HTTPFormat) SpanContextFromRequest(req *http.Request) (sc trace.SpanContext, ok bool) {
|
||
|
tid, ok := ParseTraceID(req.Header.Get(TraceIDHeader))
|
||
|
if !ok {
|
||
|
return trace.SpanContext{}, false
|
||
|
}
|
||
|
sid, ok := ParseSpanID(req.Header.Get(SpanIDHeader))
|
||
|
if !ok {
|
||
|
return trace.SpanContext{}, false
|
||
|
}
|
||
|
sampled, _ := ParseSampled(req.Header.Get(SampledHeader))
|
||
|
return trace.SpanContext{
|
||
|
TraceID: tid,
|
||
|
SpanID: sid,
|
||
|
TraceOptions: sampled,
|
||
|
}, true
|
||
|
}
|
||
|
|
||
|
// ParseTraceID parses the value of the X-B3-TraceId header.
|
||
|
func ParseTraceID(tid string) (trace.TraceID, bool) {
|
||
|
if tid == "" {
|
||
|
return trace.TraceID{}, false
|
||
|
}
|
||
|
b, err := hex.DecodeString(tid)
|
||
|
if err != nil {
|
||
|
return trace.TraceID{}, false
|
||
|
}
|
||
|
var traceID trace.TraceID
|
||
|
if len(b) <= 8 {
|
||
|
// The lower 64-bits.
|
||
|
start := 8 + (8 - len(b))
|
||
|
copy(traceID[start:], b)
|
||
|
} else {
|
||
|
start := 16 - len(b)
|
||
|
copy(traceID[start:], b)
|
||
|
}
|
||
|
|
||
|
return traceID, true
|
||
|
}
|
||
|
|
||
|
// ParseSpanID parses the value of the X-B3-SpanId or X-B3-ParentSpanId headers.
|
||
|
func ParseSpanID(sid string) (spanID trace.SpanID, ok bool) {
|
||
|
if sid == "" {
|
||
|
return trace.SpanID{}, false
|
||
|
}
|
||
|
b, err := hex.DecodeString(sid)
|
||
|
if err != nil {
|
||
|
return trace.SpanID{}, false
|
||
|
}
|
||
|
start := 8 - len(b)
|
||
|
copy(spanID[start:], b)
|
||
|
return spanID, true
|
||
|
}
|
||
|
|
||
|
// ParseSampled parses the value of the X-B3-Sampled header.
|
||
|
func ParseSampled(sampled string) (trace.TraceOptions, bool) {
|
||
|
switch sampled {
|
||
|
case "true", "1":
|
||
|
return trace.TraceOptions(1), true
|
||
|
default:
|
||
|
return trace.TraceOptions(0), false
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// SpanContextToRequest modifies the given request to include B3 headers.
|
||
|
func (f *HTTPFormat) SpanContextToRequest(sc trace.SpanContext, req *http.Request) {
|
||
|
req.Header.Set(TraceIDHeader, hex.EncodeToString(sc.TraceID[:]))
|
||
|
req.Header.Set(SpanIDHeader, hex.EncodeToString(sc.SpanID[:]))
|
||
|
|
||
|
var sampled string
|
||
|
if sc.IsSampled() {
|
||
|
sampled = "1"
|
||
|
} else {
|
||
|
sampled = "0"
|
||
|
}
|
||
|
req.Header.Set(SampledHeader, sampled)
|
||
|
}
|