| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- // Copyright The OpenTelemetry Authors
- // SPDX-License-Identifier: Apache-2.0
- package trace // import "go.opentelemetry.io/otel/trace"
- import (
- "encoding/json"
- )
- const (
- // FlagsSampled is a bitmask with the sampled bit set. A SpanContext
- // with the sampling bit set means the span is sampled.
- FlagsSampled = TraceFlags(0x01)
- errInvalidHexID errorConst = "trace-id and span-id can only contain [0-9a-f] characters, all lowercase"
- errInvalidTraceIDLength errorConst = "hex encoded trace-id must have length equals to 32"
- errNilTraceID errorConst = "trace-id can't be all zero"
- errInvalidSpanIDLength errorConst = "hex encoded span-id must have length equals to 16"
- errNilSpanID errorConst = "span-id can't be all zero"
- )
- type errorConst string
- func (e errorConst) Error() string {
- return string(e)
- }
- // TraceID is a unique identity of a trace.
- // nolint:revive // revive complains about stutter of `trace.TraceID`.
- type TraceID [16]byte
- var (
- nilTraceID TraceID
- _ json.Marshaler = nilTraceID
- )
- // IsValid reports whether the trace TraceID is valid. A valid trace ID does
- // not consist of zeros only.
- func (t TraceID) IsValid() bool {
- return t != nilTraceID
- }
- // MarshalJSON implements a custom marshal function to encode TraceID
- // as a hex string.
- func (t TraceID) MarshalJSON() ([]byte, error) {
- b := [32 + 2]byte{0: '"', 33: '"'}
- h := t.hexBytes()
- copy(b[1:], h[:])
- return b[:], nil
- }
- // String returns the hex string representation form of a TraceID.
- func (t TraceID) String() string {
- h := t.hexBytes()
- return string(h[:])
- }
- // hexBytes returns the hex string representation form of a TraceID.
- func (t TraceID) hexBytes() [32]byte {
- return [32]byte{
- hexLU[t[0x0]>>4], hexLU[t[0x0]&0xf],
- hexLU[t[0x1]>>4], hexLU[t[0x1]&0xf],
- hexLU[t[0x2]>>4], hexLU[t[0x2]&0xf],
- hexLU[t[0x3]>>4], hexLU[t[0x3]&0xf],
- hexLU[t[0x4]>>4], hexLU[t[0x4]&0xf],
- hexLU[t[0x5]>>4], hexLU[t[0x5]&0xf],
- hexLU[t[0x6]>>4], hexLU[t[0x6]&0xf],
- hexLU[t[0x7]>>4], hexLU[t[0x7]&0xf],
- hexLU[t[0x8]>>4], hexLU[t[0x8]&0xf],
- hexLU[t[0x9]>>4], hexLU[t[0x9]&0xf],
- hexLU[t[0xa]>>4], hexLU[t[0xa]&0xf],
- hexLU[t[0xb]>>4], hexLU[t[0xb]&0xf],
- hexLU[t[0xc]>>4], hexLU[t[0xc]&0xf],
- hexLU[t[0xd]>>4], hexLU[t[0xd]&0xf],
- hexLU[t[0xe]>>4], hexLU[t[0xe]&0xf],
- hexLU[t[0xf]>>4], hexLU[t[0xf]&0xf],
- }
- }
- // SpanID is a unique identity of a span in a trace.
- type SpanID [8]byte
- var (
- nilSpanID SpanID
- _ json.Marshaler = nilSpanID
- )
- // IsValid reports whether the SpanID is valid. A valid SpanID does not consist
- // of zeros only.
- func (s SpanID) IsValid() bool {
- return s != nilSpanID
- }
- // MarshalJSON implements a custom marshal function to encode SpanID
- // as a hex string.
- func (s SpanID) MarshalJSON() ([]byte, error) {
- b := [16 + 2]byte{0: '"', 17: '"'}
- h := s.hexBytes()
- copy(b[1:], h[:])
- return b[:], nil
- }
- // String returns the hex string representation form of a SpanID.
- func (s SpanID) String() string {
- b := s.hexBytes()
- return string(b[:])
- }
- func (s SpanID) hexBytes() [16]byte {
- return [16]byte{
- hexLU[s[0]>>4], hexLU[s[0]&0xf],
- hexLU[s[1]>>4], hexLU[s[1]&0xf],
- hexLU[s[2]>>4], hexLU[s[2]&0xf],
- hexLU[s[3]>>4], hexLU[s[3]&0xf],
- hexLU[s[4]>>4], hexLU[s[4]&0xf],
- hexLU[s[5]>>4], hexLU[s[5]&0xf],
- hexLU[s[6]>>4], hexLU[s[6]&0xf],
- hexLU[s[7]>>4], hexLU[s[7]&0xf],
- }
- }
- // TraceIDFromHex returns a TraceID from a hex string if it is compliant with
- // the W3C trace-context specification. See more at
- // https://www.w3.org/TR/trace-context/#trace-id
- // nolint:revive // revive complains about stutter of `trace.TraceIDFromHex`.
- func TraceIDFromHex(h string) (TraceID, error) {
- if len(h) != 32 {
- return [16]byte{}, errInvalidTraceIDLength
- }
- var b [16]byte
- invalidMark := byte(0)
- for i := 0; i < len(h); i += 4 {
- b[i/2] = (hexRev[h[i]] << 4) | hexRev[h[i+1]]
- b[i/2+1] = (hexRev[h[i+2]] << 4) | hexRev[h[i+3]]
- invalidMark |= hexRev[h[i]] | hexRev[h[i+1]] | hexRev[h[i+2]] | hexRev[h[i+3]]
- }
- // If the upper 4 bits of any byte are not zero, there was an invalid hex
- // character since invalid hex characters are 0xff in hexRev.
- if invalidMark&0xf0 != 0 {
- return [16]byte{}, errInvalidHexID
- }
- // If we didn't set any bits, then h was all zeros.
- if invalidMark == 0 {
- return [16]byte{}, errNilTraceID
- }
- return b, nil
- }
- // SpanIDFromHex returns a SpanID from a hex string if it is compliant
- // with the w3c trace-context specification.
- // See more at https://www.w3.org/TR/trace-context/#parent-id
- func SpanIDFromHex(h string) (SpanID, error) {
- if len(h) != 16 {
- return [8]byte{}, errInvalidSpanIDLength
- }
- var b [8]byte
- invalidMark := byte(0)
- for i := 0; i < len(h); i += 4 {
- b[i/2] = (hexRev[h[i]] << 4) | hexRev[h[i+1]]
- b[i/2+1] = (hexRev[h[i+2]] << 4) | hexRev[h[i+3]]
- invalidMark |= hexRev[h[i]] | hexRev[h[i+1]] | hexRev[h[i+2]] | hexRev[h[i+3]]
- }
- // If the upper 4 bits of any byte are not zero, there was an invalid hex
- // character since invalid hex characters are 0xff in hexRev.
- if invalidMark&0xf0 != 0 {
- return [8]byte{}, errInvalidHexID
- }
- // If we didn't set any bits, then h was all zeros.
- if invalidMark == 0 {
- return [8]byte{}, errNilSpanID
- }
- return b, nil
- }
- // TraceFlags contains flags that can be set on a SpanContext.
- type TraceFlags byte //nolint:revive // revive complains about stutter of `trace.TraceFlags`.
- // IsSampled reports whether the sampling bit is set in the TraceFlags.
- func (tf TraceFlags) IsSampled() bool {
- return tf&FlagsSampled == FlagsSampled
- }
- // WithSampled sets the sampling bit in a new copy of the TraceFlags.
- func (tf TraceFlags) WithSampled(sampled bool) TraceFlags { // nolint:revive // sampled is not a control flag.
- if sampled {
- return tf | FlagsSampled
- }
- return tf &^ FlagsSampled
- }
- // MarshalJSON implements a custom marshal function to encode TraceFlags
- // as a hex string.
- func (tf TraceFlags) MarshalJSON() ([]byte, error) {
- b := [2 + 2]byte{0: '"', 3: '"'}
- h := tf.hexBytes()
- copy(b[1:], h[:])
- return b[:], nil
- }
- // String returns the hex string representation form of TraceFlags.
- func (tf TraceFlags) String() string {
- h := tf.hexBytes()
- return string(h[:])
- }
- func (tf TraceFlags) hexBytes() [2]byte {
- return [2]byte{hexLU[tf>>4], hexLU[tf&0xf]}
- }
- // SpanContextConfig contains mutable fields usable for constructing
- // an immutable SpanContext.
- type SpanContextConfig struct {
- TraceID TraceID
- SpanID SpanID
- TraceFlags TraceFlags
- TraceState TraceState
- Remote bool
- }
- // NewSpanContext constructs a SpanContext using values from the provided
- // SpanContextConfig.
- func NewSpanContext(config SpanContextConfig) SpanContext {
- return SpanContext{
- traceID: config.TraceID,
- spanID: config.SpanID,
- traceFlags: config.TraceFlags,
- traceState: config.TraceState,
- remote: config.Remote,
- }
- }
- // SpanContext contains identifying trace information about a Span.
- type SpanContext struct {
- traceID TraceID
- spanID SpanID
- traceFlags TraceFlags
- traceState TraceState
- remote bool
- }
- var _ json.Marshaler = SpanContext{}
- // IsValid reports whether the SpanContext is valid. A valid span context has a
- // valid TraceID and SpanID.
- func (sc SpanContext) IsValid() bool {
- return sc.HasTraceID() && sc.HasSpanID()
- }
- // IsRemote reports whether the SpanContext represents a remotely-created Span.
- func (sc SpanContext) IsRemote() bool {
- return sc.remote
- }
- // WithRemote returns a copy of sc with the Remote property set to remote.
- func (sc SpanContext) WithRemote(remote bool) SpanContext {
- return SpanContext{
- traceID: sc.traceID,
- spanID: sc.spanID,
- traceFlags: sc.traceFlags,
- traceState: sc.traceState,
- remote: remote,
- }
- }
- // TraceID returns the TraceID from the SpanContext.
- func (sc SpanContext) TraceID() TraceID {
- return sc.traceID
- }
- // HasTraceID reports whether the SpanContext has a valid TraceID.
- func (sc SpanContext) HasTraceID() bool {
- return sc.traceID.IsValid()
- }
- // WithTraceID returns a new SpanContext with the TraceID replaced.
- func (sc SpanContext) WithTraceID(traceID TraceID) SpanContext {
- return SpanContext{
- traceID: traceID,
- spanID: sc.spanID,
- traceFlags: sc.traceFlags,
- traceState: sc.traceState,
- remote: sc.remote,
- }
- }
- // SpanID returns the SpanID from the SpanContext.
- func (sc SpanContext) SpanID() SpanID {
- return sc.spanID
- }
- // HasSpanID reports whether the SpanContext has a valid SpanID.
- func (sc SpanContext) HasSpanID() bool {
- return sc.spanID.IsValid()
- }
- // WithSpanID returns a new SpanContext with the SpanID replaced.
- func (sc SpanContext) WithSpanID(spanID SpanID) SpanContext {
- return SpanContext{
- traceID: sc.traceID,
- spanID: spanID,
- traceFlags: sc.traceFlags,
- traceState: sc.traceState,
- remote: sc.remote,
- }
- }
- // TraceFlags returns the flags from the SpanContext.
- func (sc SpanContext) TraceFlags() TraceFlags {
- return sc.traceFlags
- }
- // IsSampled reports whether the sampling bit is set in the SpanContext's TraceFlags.
- func (sc SpanContext) IsSampled() bool {
- return sc.traceFlags.IsSampled()
- }
- // WithTraceFlags returns a new SpanContext with the TraceFlags replaced.
- func (sc SpanContext) WithTraceFlags(flags TraceFlags) SpanContext {
- return SpanContext{
- traceID: sc.traceID,
- spanID: sc.spanID,
- traceFlags: flags,
- traceState: sc.traceState,
- remote: sc.remote,
- }
- }
- // TraceState returns the TraceState from the SpanContext.
- func (sc SpanContext) TraceState() TraceState {
- return sc.traceState
- }
- // WithTraceState returns a new SpanContext with the TraceState replaced.
- func (sc SpanContext) WithTraceState(state TraceState) SpanContext {
- return SpanContext{
- traceID: sc.traceID,
- spanID: sc.spanID,
- traceFlags: sc.traceFlags,
- traceState: state,
- remote: sc.remote,
- }
- }
- // Equal reports whether two SpanContext values are equal.
- func (sc SpanContext) Equal(other SpanContext) bool {
- return sc.traceID == other.traceID &&
- sc.spanID == other.spanID &&
- sc.traceFlags == other.traceFlags &&
- sc.traceState.String() == other.traceState.String() &&
- sc.remote == other.remote
- }
- // MarshalJSON implements a custom marshal function to encode a SpanContext.
- func (sc SpanContext) MarshalJSON() ([]byte, error) {
- return json.Marshal(SpanContextConfig{
- TraceID: sc.traceID,
- SpanID: sc.spanID,
- TraceFlags: sc.traceFlags,
- TraceState: sc.traceState,
- Remote: sc.remote,
- })
- }
|