mirror of
https://github.com/cloudflare/cloudflared.git
synced 2025-07-28 07:39:58 +00:00
121 lines
3.7 KiB
Go
121 lines
3.7 KiB
Go
package mocktracer
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/opentracing/opentracing-go"
|
|
)
|
|
|
|
const mockTextMapIdsPrefix = "mockpfx-ids-"
|
|
const mockTextMapBaggagePrefix = "mockpfx-baggage-"
|
|
|
|
var emptyContext = MockSpanContext{}
|
|
|
|
// Injector is responsible for injecting SpanContext instances in a manner suitable
|
|
// for propagation via a format-specific "carrier" object. Typically the
|
|
// injection will take place across an RPC boundary, but message queues and
|
|
// other IPC mechanisms are also reasonable places to use an Injector.
|
|
type Injector interface {
|
|
// Inject takes `SpanContext` and injects it into `carrier`. The actual type
|
|
// of `carrier` depends on the `format` passed to `Tracer.Inject()`.
|
|
//
|
|
// Implementations may return opentracing.ErrInvalidCarrier or any other
|
|
// implementation-specific error if injection fails.
|
|
Inject(ctx MockSpanContext, carrier interface{}) error
|
|
}
|
|
|
|
// Extractor is responsible for extracting SpanContext instances from a
|
|
// format-specific "carrier" object. Typically the extraction will take place
|
|
// on the server side of an RPC boundary, but message queues and other IPC
|
|
// mechanisms are also reasonable places to use an Extractor.
|
|
type Extractor interface {
|
|
// Extract decodes a SpanContext instance from the given `carrier`,
|
|
// or (nil, opentracing.ErrSpanContextNotFound) if no context could
|
|
// be found in the `carrier`.
|
|
Extract(carrier interface{}) (MockSpanContext, error)
|
|
}
|
|
|
|
// TextMapPropagator implements Injector/Extractor for TextMap and HTTPHeaders formats.
|
|
type TextMapPropagator struct {
|
|
HTTPHeaders bool
|
|
}
|
|
|
|
// Inject implements the Injector interface
|
|
func (t *TextMapPropagator) Inject(spanContext MockSpanContext, carrier interface{}) error {
|
|
writer, ok := carrier.(opentracing.TextMapWriter)
|
|
if !ok {
|
|
return opentracing.ErrInvalidCarrier
|
|
}
|
|
// Ids:
|
|
writer.Set(mockTextMapIdsPrefix+"traceid", strconv.Itoa(spanContext.TraceID))
|
|
writer.Set(mockTextMapIdsPrefix+"spanid", strconv.Itoa(spanContext.SpanID))
|
|
writer.Set(mockTextMapIdsPrefix+"sampled", fmt.Sprint(spanContext.Sampled))
|
|
// Baggage:
|
|
for baggageKey, baggageVal := range spanContext.Baggage {
|
|
safeVal := baggageVal
|
|
if t.HTTPHeaders {
|
|
safeVal = url.QueryEscape(baggageVal)
|
|
}
|
|
writer.Set(mockTextMapBaggagePrefix+baggageKey, safeVal)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Extract implements the Extractor interface
|
|
func (t *TextMapPropagator) Extract(carrier interface{}) (MockSpanContext, error) {
|
|
reader, ok := carrier.(opentracing.TextMapReader)
|
|
if !ok {
|
|
return emptyContext, opentracing.ErrInvalidCarrier
|
|
}
|
|
rval := MockSpanContext{0, 0, true, nil}
|
|
err := reader.ForeachKey(func(key, val string) error {
|
|
lowerKey := strings.ToLower(key)
|
|
switch {
|
|
case lowerKey == mockTextMapIdsPrefix+"traceid":
|
|
// Ids:
|
|
i, err := strconv.Atoi(val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rval.TraceID = i
|
|
case lowerKey == mockTextMapIdsPrefix+"spanid":
|
|
// Ids:
|
|
i, err := strconv.Atoi(val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rval.SpanID = i
|
|
case lowerKey == mockTextMapIdsPrefix+"sampled":
|
|
b, err := strconv.ParseBool(val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rval.Sampled = b
|
|
case strings.HasPrefix(lowerKey, mockTextMapBaggagePrefix):
|
|
// Baggage:
|
|
if rval.Baggage == nil {
|
|
rval.Baggage = make(map[string]string)
|
|
}
|
|
safeVal := val
|
|
if t.HTTPHeaders {
|
|
// unescape errors are ignored, nothing can be done
|
|
if rawVal, err := url.QueryUnescape(val); err == nil {
|
|
safeVal = rawVal
|
|
}
|
|
}
|
|
rval.Baggage[lowerKey[len(mockTextMapBaggagePrefix):]] = safeVal
|
|
}
|
|
return nil
|
|
})
|
|
if rval.TraceID == 0 || rval.SpanID == 0 {
|
|
return emptyContext, opentracing.ErrSpanContextNotFound
|
|
}
|
|
if err != nil {
|
|
return emptyContext, err
|
|
}
|
|
return rval, nil
|
|
}
|