forked from TrueCloudLab/lego
187 lines
5.3 KiB
Go
187 lines
5.3 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 tracecontext contains HTTP propagator for TraceContext standard.
|
|
// See https://github.com/w3c/distributed-tracing for more information.
|
|
package tracecontext // import "go.opencensus.io/plugin/ochttp/propagation/tracecontext"
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"net/http"
|
|
"net/textproto"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"go.opencensus.io/trace"
|
|
"go.opencensus.io/trace/propagation"
|
|
"go.opencensus.io/trace/tracestate"
|
|
)
|
|
|
|
const (
|
|
supportedVersion = 0
|
|
maxVersion = 254
|
|
maxTracestateLen = 512
|
|
traceparentHeader = "traceparent"
|
|
tracestateHeader = "tracestate"
|
|
trimOWSRegexFmt = `^[\x09\x20]*(.*[^\x20\x09])[\x09\x20]*$`
|
|
)
|
|
|
|
var trimOWSRegExp = regexp.MustCompile(trimOWSRegexFmt)
|
|
|
|
var _ propagation.HTTPFormat = (*HTTPFormat)(nil)
|
|
|
|
// HTTPFormat implements the TraceContext trace propagation format.
|
|
type HTTPFormat struct{}
|
|
|
|
// SpanContextFromRequest extracts a span context from incoming requests.
|
|
func (f *HTTPFormat) SpanContextFromRequest(req *http.Request) (sc trace.SpanContext, ok bool) {
|
|
h, ok := getRequestHeader(req, traceparentHeader, false)
|
|
if !ok {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
sections := strings.Split(h, "-")
|
|
if len(sections) < 4 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
|
|
if len(sections[0]) != 2 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
ver, err := hex.DecodeString(sections[0])
|
|
if err != nil {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
version := int(ver[0])
|
|
if version > maxVersion {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
|
|
if version == 0 && len(sections) != 4 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
|
|
if len(sections[1]) != 32 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
tid, err := hex.DecodeString(sections[1])
|
|
if err != nil {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
copy(sc.TraceID[:], tid)
|
|
|
|
if len(sections[2]) != 16 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
sid, err := hex.DecodeString(sections[2])
|
|
if err != nil {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
copy(sc.SpanID[:], sid)
|
|
|
|
opts, err := hex.DecodeString(sections[3])
|
|
if err != nil || len(opts) < 1 {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
sc.TraceOptions = trace.TraceOptions(opts[0])
|
|
|
|
// Don't allow all zero trace or span ID.
|
|
if sc.TraceID == [16]byte{} || sc.SpanID == [8]byte{} {
|
|
return trace.SpanContext{}, false
|
|
}
|
|
|
|
sc.Tracestate = tracestateFromRequest(req)
|
|
return sc, true
|
|
}
|
|
|
|
// getRequestHeader returns a combined header field according to RFC7230 section 3.2.2.
|
|
// If commaSeparated is true, multiple header fields with the same field name using be
|
|
// combined using ",".
|
|
// If no header was found using the given name, "ok" would be false.
|
|
// If more than one headers was found using the given name, while commaSeparated is false,
|
|
// "ok" would be false.
|
|
func getRequestHeader(req *http.Request, name string, commaSeparated bool) (hdr string, ok bool) {
|
|
v := req.Header[textproto.CanonicalMIMEHeaderKey(name)]
|
|
switch len(v) {
|
|
case 0:
|
|
return "", false
|
|
case 1:
|
|
return v[0], true
|
|
default:
|
|
return strings.Join(v, ","), commaSeparated
|
|
}
|
|
}
|
|
|
|
// TODO(rghetia): return an empty Tracestate when parsing tracestate header encounters an error.
|
|
// Revisit to return additional boolean value to indicate parsing error when following issues
|
|
// are resolved.
|
|
// https://github.com/w3c/distributed-tracing/issues/172
|
|
// https://github.com/w3c/distributed-tracing/issues/175
|
|
func tracestateFromRequest(req *http.Request) *tracestate.Tracestate {
|
|
h, _ := getRequestHeader(req, tracestateHeader, true)
|
|
if h == "" {
|
|
return nil
|
|
}
|
|
|
|
var entries []tracestate.Entry
|
|
pairs := strings.Split(h, ",")
|
|
hdrLenWithoutOWS := len(pairs) - 1 // Number of commas
|
|
for _, pair := range pairs {
|
|
matches := trimOWSRegExp.FindStringSubmatch(pair)
|
|
if matches == nil {
|
|
return nil
|
|
}
|
|
pair = matches[1]
|
|
hdrLenWithoutOWS += len(pair)
|
|
if hdrLenWithoutOWS > maxTracestateLen {
|
|
return nil
|
|
}
|
|
kv := strings.Split(pair, "=")
|
|
if len(kv) != 2 {
|
|
return nil
|
|
}
|
|
entries = append(entries, tracestate.Entry{Key: kv[0], Value: kv[1]})
|
|
}
|
|
ts, err := tracestate.New(nil, entries...)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
|
|
return ts
|
|
}
|
|
|
|
func tracestateToRequest(sc trace.SpanContext, req *http.Request) {
|
|
var pairs = make([]string, 0, len(sc.Tracestate.Entries()))
|
|
if sc.Tracestate != nil {
|
|
for _, entry := range sc.Tracestate.Entries() {
|
|
pairs = append(pairs, strings.Join([]string{entry.Key, entry.Value}, "="))
|
|
}
|
|
h := strings.Join(pairs, ",")
|
|
|
|
if h != "" && len(h) <= maxTracestateLen {
|
|
req.Header.Set(tracestateHeader, h)
|
|
}
|
|
}
|
|
}
|
|
|
|
// SpanContextToRequest modifies the given request to include traceparent and tracestate headers.
|
|
func (f *HTTPFormat) SpanContextToRequest(sc trace.SpanContext, req *http.Request) {
|
|
h := fmt.Sprintf("%x-%x-%x-%x",
|
|
[]byte{supportedVersion},
|
|
sc.TraceID[:],
|
|
sc.SpanID[:],
|
|
[]byte{byte(sc.TraceOptions)})
|
|
req.Header.Set(traceparentHeader, h)
|
|
tracestateToRequest(sc, req)
|
|
}
|