mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-03 22:52:30 +02:00
Unify API Span Start/End Options (#1108)
* Unify API Span Start/End Options Replace both with `SpanOption`. Add a unified `SpanConfig` to match and a `SpanConfigure` function to parse a `SpanConfig` from `SpanOption`s. Update all the related options to use new `SpanOption`s. * No non-zero SpanConfig defaults The SDK uses an internal clock for the current time that cannot be use if it does not know the time has not been set. * Append attributes for WithAttributes This preserves existing behavior. * Add unit test for SpanConfigure * Propagate changes * Update append option documentation * Update testing comments * Move comments on guarantees to appropriate function * Add documentation for SDK methods Include SDK implementation specific information in the Tracer Start method and Span End method. * Add changes to Changelog * Apply suggestions from code review Co-authored-by: ET <evantorrie@users.noreply.github.com> * Update the SpanKind comment in the SpanConfig Try for a less tautological comment. Co-authored-by: ET <evantorrie@users.noreply.github.com>
This commit is contained in:
parent
4d83d5b571
commit
d143b8fbf8
@ -10,6 +10,7 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm
|
|||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
- A `SpanConfigure` function in `go.opentelemetry.io/otel/api/trace` to create a new `SpanConfig` from `SpanOption`s. (#1108)
|
||||||
- In the `go.opentelemetry.io/otel/api/trace` package a new `TracerConfigure` function was added to configure a new `TracerConfig`.
|
- In the `go.opentelemetry.io/otel/api/trace` package a new `TracerConfigure` function was added to configure a new `TracerConfig`.
|
||||||
This addition was made to conform with our project option conventions. (#1109)
|
This addition was made to conform with our project option conventions. (#1109)
|
||||||
- Instrumentation library information was added to the Zipkin exporter. (#1119)
|
- Instrumentation library information was added to the Zipkin exporter. (#1119)
|
||||||
@ -19,6 +20,10 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm
|
|||||||
- Add reconnecting udp connection type to Jaeger exporter.
|
- Add reconnecting udp connection type to Jaeger exporter.
|
||||||
This change adds a new optional implementation of the udp conn interface used to detect changes to an agent's host dns record.
|
This change adds a new optional implementation of the udp conn interface used to detect changes to an agent's host dns record.
|
||||||
It then adopts the new destination address to ensure the exporter doesn't get stuck. This change was ported from jaegertracing/jaeger-client-go#520. (#1063)
|
It then adopts the new destination address to ensure the exporter doesn't get stuck. This change was ported from jaegertracing/jaeger-client-go#520. (#1063)
|
||||||
|
- Replace `StartOption` and `EndOption` in `go.opentelemetry.io/otel/api/trace` with `SpanOption`.
|
||||||
|
This change is matched by replacing the `StartConfig` and `EndConfig` with a unified `SpanConfig`. (#1108)
|
||||||
|
- Replace the `LinkedTo` span option in `go.opentelemetry.io/otel/api/trace` with `WithLinks`.
|
||||||
|
This is be more consistent with our other option patterns, i.e. passing the item to be configured directly instead of its component parts, and provides a cleaner function signature. (#1108)
|
||||||
- The `go.opentelemetry.io/otel/api/trace` `TracerOption` was changed to an interface to conform to project option conventions. (#1109)
|
- The `go.opentelemetry.io/otel/api/trace` `TracerOption` was changed to an interface to conform to project option conventions. (#1109)
|
||||||
- Rename Jaeger tags used for instrumentation library information to reflect changes in OpenTelemetry specification. (#1119)
|
- Rename Jaeger tags used for instrumentation library information to reflect changes in OpenTelemetry specification. (#1119)
|
||||||
|
|
||||||
|
@ -116,7 +116,7 @@ func (t *tracer) setDelegate(provider trace.Provider) {
|
|||||||
|
|
||||||
// Start implements trace.Tracer by forwarding the call to t.delegate if
|
// Start implements trace.Tracer by forwarding the call to t.delegate if
|
||||||
// set, otherwise it forwards the call to a NoopTracer.
|
// set, otherwise it forwards the call to a NoopTracer.
|
||||||
func (t *tracer) Start(ctx context.Context, name string, opts ...trace.StartOption) (context.Context, trace.Span) {
|
func (t *tracer) Start(ctx context.Context, name string, opts ...trace.SpanOption) (context.Context, trace.Span) {
|
||||||
if t.delegate != nil {
|
if t.delegate != nil {
|
||||||
return t.delegate.Start(ctx, name, opts...)
|
return t.delegate.Start(ctx, name, opts...)
|
||||||
}
|
}
|
||||||
|
182
api/trace/api.go
182
api/trace/api.go
@ -68,23 +68,7 @@ func WithInstrumentationVersion(version string) TracerOption {
|
|||||||
|
|
||||||
type Tracer interface {
|
type Tracer interface {
|
||||||
// Start a span.
|
// Start a span.
|
||||||
Start(ctx context.Context, spanName string, opts ...StartOption) (context.Context, Span)
|
Start(ctx context.Context, spanName string, opts ...SpanOption) (context.Context, Span)
|
||||||
}
|
|
||||||
|
|
||||||
// EndConfig provides options to set properties of span at the time of ending
|
|
||||||
// the span.
|
|
||||||
type EndConfig struct {
|
|
||||||
EndTime time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
// EndOption applies changes to EndConfig that sets options when the span is ended.
|
|
||||||
type EndOption func(*EndConfig)
|
|
||||||
|
|
||||||
// WithEndTime sets the end time of the span to provided time t, when it is ended.
|
|
||||||
func WithEndTime(t time.Time) EndOption {
|
|
||||||
return func(c *EndConfig) {
|
|
||||||
c.EndTime = t
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ErrorConfig provides options to set properties of an error event at the time it is recorded.
|
// ErrorConfig provides options to set properties of an error event at the time it is recorded.
|
||||||
@ -116,7 +100,7 @@ type Span interface {
|
|||||||
|
|
||||||
// End completes the span. No updates are allowed to span after it
|
// End completes the span. No updates are allowed to span after it
|
||||||
// ends. The only exception is setting status of the span.
|
// ends. The only exception is setting status of the span.
|
||||||
End(options ...EndOption)
|
End(options ...SpanOption)
|
||||||
|
|
||||||
// AddEvent adds an event to the span.
|
// AddEvent adds an event to the span.
|
||||||
AddEvent(ctx context.Context, name string, attrs ...label.KeyValue)
|
AddEvent(ctx context.Context, name string, attrs ...label.KeyValue)
|
||||||
@ -153,18 +137,106 @@ type Span interface {
|
|||||||
SetAttribute(string, interface{})
|
SetAttribute(string, interface{})
|
||||||
}
|
}
|
||||||
|
|
||||||
// StartOption applies changes to StartConfig that sets options at span start time.
|
// SpanConfig is a group of options for a Span.
|
||||||
type StartOption func(*StartConfig)
|
type SpanConfig struct {
|
||||||
|
// Attributes describe the associated qualities of a Span.
|
||||||
// StartConfig provides options to set properties of span at the time of starting
|
|
||||||
// a new span.
|
|
||||||
type StartConfig struct {
|
|
||||||
Attributes []label.KeyValue
|
Attributes []label.KeyValue
|
||||||
StartTime time.Time
|
// Timestamp is a time in a Span life-cycle.
|
||||||
Links []Link
|
Timestamp time.Time
|
||||||
Record bool
|
// Links are the associations a Span has with other Spans.
|
||||||
NewRoot bool
|
Links []Link
|
||||||
SpanKind SpanKind
|
// Record is the recording state of a Span.
|
||||||
|
Record bool
|
||||||
|
// NewRoot identifies a Span as the root Span for a new trace. This is
|
||||||
|
// commonly used when an existing trace crosses trust boundaries and the
|
||||||
|
// remote parent span context should be ignored for security.
|
||||||
|
NewRoot bool
|
||||||
|
// SpanKind is the role a Span has in a trace.
|
||||||
|
SpanKind SpanKind
|
||||||
|
}
|
||||||
|
|
||||||
|
// SpanConfigure applies all the options to a returned SpanConfig.
|
||||||
|
// The default value for all the fields of the returned SpanConfig are the
|
||||||
|
// default zero value of the type. Also, this does not perform any validation
|
||||||
|
// on the returned SpanConfig (e.g. no uniqueness checking or bounding of
|
||||||
|
// data). Instead, it is left to the implementations of the SDK to perform this
|
||||||
|
// action.
|
||||||
|
func SpanConfigure(options []SpanOption) *SpanConfig {
|
||||||
|
config := new(SpanConfig)
|
||||||
|
for _, option := range options {
|
||||||
|
option.Apply(config)
|
||||||
|
}
|
||||||
|
return config
|
||||||
|
}
|
||||||
|
|
||||||
|
// SpanOption applies an option to a SpanConfig.
|
||||||
|
type SpanOption interface {
|
||||||
|
Apply(*SpanConfig)
|
||||||
|
}
|
||||||
|
|
||||||
|
type attributeSpanOption []label.KeyValue
|
||||||
|
|
||||||
|
func (o attributeSpanOption) Apply(c *SpanConfig) {
|
||||||
|
c.Attributes = append(c.Attributes, []label.KeyValue(o)...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithAttributes adds the attributes to a span. These attributes are meant to
|
||||||
|
// provide additional information about the work the Span represents. The
|
||||||
|
// attributes are added to the existing Span attributes, i.e. this does not
|
||||||
|
// overwrite.
|
||||||
|
func WithAttributes(attributes ...label.KeyValue) SpanOption {
|
||||||
|
return attributeSpanOption(attributes)
|
||||||
|
}
|
||||||
|
|
||||||
|
type timestampSpanOption time.Time
|
||||||
|
|
||||||
|
func (o timestampSpanOption) Apply(c *SpanConfig) { c.Timestamp = time.Time(o) }
|
||||||
|
|
||||||
|
// WithTimestamp sets the time of a Span life-cycle moment (e.g. started or
|
||||||
|
// stopped).
|
||||||
|
func WithTimestamp(t time.Time) SpanOption {
|
||||||
|
return timestampSpanOption(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
type linksSpanOption []Link
|
||||||
|
|
||||||
|
func (o linksSpanOption) Apply(c *SpanConfig) { c.Links = append(c.Links, []Link(o)...) }
|
||||||
|
|
||||||
|
// WithLinks adds links to a Span. The links are added to the existing Span
|
||||||
|
// links, i.e. this does not overwrite.
|
||||||
|
func WithLinks(links ...Link) SpanOption {
|
||||||
|
return linksSpanOption(links)
|
||||||
|
}
|
||||||
|
|
||||||
|
type recordSpanOption bool
|
||||||
|
|
||||||
|
func (o recordSpanOption) Apply(c *SpanConfig) { c.Record = bool(o) }
|
||||||
|
|
||||||
|
// WithRecord specifies that the span should be recorded. It is important to
|
||||||
|
// note that implementations may override this option, i.e. if the span is a
|
||||||
|
// child of an un-sampled trace.
|
||||||
|
func WithRecord() SpanOption {
|
||||||
|
return recordSpanOption(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
type newRootSpanOption bool
|
||||||
|
|
||||||
|
func (o newRootSpanOption) Apply(c *SpanConfig) { c.NewRoot = bool(o) }
|
||||||
|
|
||||||
|
// WithNewRoot specifies that the Span should be treated as a root Span. Any
|
||||||
|
// existing parent span context will be ignored when defining the Span's trace
|
||||||
|
// identifiers.
|
||||||
|
func WithNewRoot() SpanOption {
|
||||||
|
return newRootSpanOption(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
type spanKindSpanOption SpanKind
|
||||||
|
|
||||||
|
func (o spanKindSpanOption) Apply(c *SpanConfig) { c.SpanKind = SpanKind(o) }
|
||||||
|
|
||||||
|
// WithSpanKind sets the SpanKind of a Span.
|
||||||
|
func WithSpanKind(kind SpanKind) SpanOption {
|
||||||
|
return spanKindSpanOption(kind)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Link is used to establish relationship between two spans within the same Trace or
|
// Link is used to establish relationship between two spans within the same Trace or
|
||||||
@ -235,55 +307,3 @@ func (sk SpanKind) String() string {
|
|||||||
return "unspecified"
|
return "unspecified"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// WithStartTime sets the start time of the span to provided time t, when it is started.
|
|
||||||
// In absence of this option, wall clock time is used as start time.
|
|
||||||
// This option is typically used when starting of the span is delayed.
|
|
||||||
func WithStartTime(t time.Time) StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.StartTime = t
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithAttributes sets attributes to span. These attributes provides additional
|
|
||||||
// data about the span.
|
|
||||||
// Multiple `WithAttributes` options appends the attributes preserving the order.
|
|
||||||
func WithAttributes(attrs ...label.KeyValue) StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.Attributes = append(c.Attributes, attrs...)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithRecord specifies that the span should be recorded.
|
|
||||||
// Note that the implementation may still override this preference,
|
|
||||||
// e.g., if the span is a child in an unsampled trace.
|
|
||||||
func WithRecord() StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.Record = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithNewRoot specifies that the current span or remote span context
|
|
||||||
// in context passed to `Start` should be ignored when deciding about
|
|
||||||
// a parent, which effectively means creating a span with new trace
|
|
||||||
// ID. The current span and the remote span context may be added as
|
|
||||||
// links to the span by the implementation.
|
|
||||||
func WithNewRoot() StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.NewRoot = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// LinkedTo allows instantiating a Span with initial Links.
|
|
||||||
func LinkedTo(sc SpanContext, attrs ...label.KeyValue) StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.Links = append(c.Links, Link{sc, attrs})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithSpanKind specifies the role a Span on a Trace.
|
|
||||||
func WithSpanKind(sk SpanKind) StartOption {
|
|
||||||
return func(c *StartConfig) {
|
|
||||||
c.SpanKind = sk
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -101,7 +101,7 @@ func (mockSpan) SetAttribute(k string, v interface{}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// End does nothing.
|
// End does nothing.
|
||||||
func (mockSpan) End(options ...trace.EndOption) {
|
func (mockSpan) End(options ...trace.SpanOption) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RecordError does nothing.
|
// RecordError does nothing.
|
||||||
|
@ -54,7 +54,7 @@ func (NoopSpan) SetAttribute(k string, v interface{}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// End does nothing.
|
// End does nothing.
|
||||||
func (NoopSpan) End(options ...EndOption) {
|
func (NoopSpan) End(options ...SpanOption) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RecordError does nothing.
|
// RecordError does nothing.
|
||||||
|
@ -23,7 +23,7 @@ type NoopTracer struct{}
|
|||||||
var _ Tracer = NoopTracer{}
|
var _ Tracer = NoopTracer{}
|
||||||
|
|
||||||
// Start starts a noop span.
|
// Start starts a noop span.
|
||||||
func (NoopTracer) Start(ctx context.Context, name string, opts ...StartOption) (context.Context, Span) {
|
func (NoopTracer) Start(ctx context.Context, name string, opts ...SpanOption) (context.Context, Span) {
|
||||||
span := NoopSpan{}
|
span := NoopSpan{}
|
||||||
return ContextWithSpan(ctx, span), span
|
return ContextWithSpan(ctx, span), span
|
||||||
}
|
}
|
||||||
|
195
api/trace/span_test.go
Normal file
195
api/trace/span_test.go
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
// Copyright The 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 trace
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
|
"go.opentelemetry.io/otel/label"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSpanConfigure(t *testing.T) {
|
||||||
|
k1v1 := label.String("key1", "value1")
|
||||||
|
k1v2 := label.String("key1", "value2")
|
||||||
|
k2v2 := label.String("key2", "value2")
|
||||||
|
|
||||||
|
timestamp0 := time.Unix(0, 0)
|
||||||
|
timestamp1 := time.Unix(0, 0)
|
||||||
|
|
||||||
|
link1 := Link{
|
||||||
|
SpanContext: SpanContext{TraceID: ID([16]byte{1, 1}), SpanID: SpanID{3}},
|
||||||
|
Attributes: []label.KeyValue{k1v1},
|
||||||
|
}
|
||||||
|
link2 := Link{
|
||||||
|
SpanContext: SpanContext{TraceID: ID([16]byte{1, 1}), SpanID: SpanID{3}},
|
||||||
|
Attributes: []label.KeyValue{k1v2, k2v2},
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
options []SpanOption
|
||||||
|
expected *SpanConfig
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// No non-zero-values should be set.
|
||||||
|
[]SpanOption{},
|
||||||
|
new(SpanConfig),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithAttributes(k1v1),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Attributes: []label.KeyValue{k1v1},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// Multiple calls should append not overwrite.
|
||||||
|
[]SpanOption{
|
||||||
|
WithAttributes(k1v1),
|
||||||
|
WithAttributes(k1v2),
|
||||||
|
WithAttributes(k2v2),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
// No uniqueness is guaranteed by the API.
|
||||||
|
Attributes: []label.KeyValue{k1v1, k1v2, k2v2},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithAttributes(k1v1, k1v2, k2v2),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
// No uniqueness is guaranteed by the API.
|
||||||
|
Attributes: []label.KeyValue{k1v1, k1v2, k2v2},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithTimestamp(timestamp0),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Timestamp: timestamp0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
// Multiple calls overwrites with last-one-wins.
|
||||||
|
WithTimestamp(timestamp0),
|
||||||
|
WithTimestamp(timestamp1),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Timestamp: timestamp1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithLinks(link1),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Links: []Link{link1},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
// Multiple calls should append not overwrite.
|
||||||
|
WithLinks(link1),
|
||||||
|
WithLinks(link1, link2),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
// No uniqueness is guaranteed by the API.
|
||||||
|
Links: []Link{link1, link1, link2},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithRecord(),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Record: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
// Multiple calls should not change Record state.
|
||||||
|
WithRecord(),
|
||||||
|
WithRecord(),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Record: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithNewRoot(),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
NewRoot: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
// Multiple calls should not change NewRoot state.
|
||||||
|
WithNewRoot(),
|
||||||
|
WithNewRoot(),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
NewRoot: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
WithSpanKind(SpanKindConsumer),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
SpanKind: SpanKindConsumer,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
[]SpanOption{
|
||||||
|
// Multiple calls overwrites with last-one-wins.
|
||||||
|
WithSpanKind(SpanKindClient),
|
||||||
|
WithSpanKind(SpanKindConsumer),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
SpanKind: SpanKindConsumer,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// Everything should work together.
|
||||||
|
[]SpanOption{
|
||||||
|
WithAttributes(k1v1),
|
||||||
|
WithTimestamp(timestamp0),
|
||||||
|
WithLinks(link1, link2),
|
||||||
|
WithRecord(),
|
||||||
|
WithNewRoot(),
|
||||||
|
WithSpanKind(SpanKindConsumer),
|
||||||
|
},
|
||||||
|
&SpanConfig{
|
||||||
|
Attributes: []label.KeyValue{k1v1},
|
||||||
|
Timestamp: timestamp0,
|
||||||
|
Links: []Link{link1, link2},
|
||||||
|
Record: true,
|
||||||
|
NewRoot: true,
|
||||||
|
SpanKind: SpanKindConsumer,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, test := range tests {
|
||||||
|
assert.Equal(t, test.expected, SpanConfigure(test.options))
|
||||||
|
}
|
||||||
|
}
|
@ -67,7 +67,7 @@ func (ms *MockSpan) SetAttribute(k string, v interface{}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// End does nothing.
|
// End does nothing.
|
||||||
func (ms *MockSpan) End(options ...apitrace.EndOption) {
|
func (ms *MockSpan) End(options ...apitrace.SpanOption) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RecordError does nothing.
|
// RecordError does nothing.
|
||||||
|
@ -47,15 +47,12 @@ var _ apitrace.Tracer = (*MockTracer)(nil)
|
|||||||
// TraceID is used from Parent Span Context and SpanID is assigned.
|
// TraceID is used from Parent Span Context and SpanID is assigned.
|
||||||
// If Parent SpanContext option is not specified then random TraceID is used.
|
// If Parent SpanContext option is not specified then random TraceID is used.
|
||||||
// No other options are supported.
|
// No other options are supported.
|
||||||
func (mt *MockTracer) Start(ctx context.Context, name string, o ...apitrace.StartOption) (context.Context, apitrace.Span) {
|
func (mt *MockTracer) Start(ctx context.Context, name string, o ...apitrace.SpanOption) (context.Context, apitrace.Span) {
|
||||||
var opts apitrace.StartConfig
|
config := apitrace.SpanConfigure(o)
|
||||||
for _, op := range o {
|
|
||||||
op(&opts)
|
|
||||||
}
|
|
||||||
var span *MockSpan
|
var span *MockSpan
|
||||||
var sc apitrace.SpanContext
|
var sc apitrace.SpanContext
|
||||||
|
|
||||||
parentSpanContext, _, _ := otelparent.GetSpanContextAndLinks(ctx, opts.NewRoot)
|
parentSpanContext, _, _ := otelparent.GetSpanContextAndLinks(ctx, config.NewRoot)
|
||||||
|
|
||||||
if !parentSpanContext.IsValid() {
|
if !parentSpanContext.IsValid() {
|
||||||
sc = apitrace.SpanContext{}
|
sc = apitrace.SpanContext{}
|
||||||
|
@ -55,7 +55,7 @@ func (s *Span) Tracer() trace.Tracer {
|
|||||||
return s.tracer
|
return s.tracer
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Span) End(opts ...trace.EndOption) {
|
func (s *Span) End(opts ...trace.SpanOption) {
|
||||||
s.lock.Lock()
|
s.lock.Lock()
|
||||||
defer s.lock.Unlock()
|
defer s.lock.Unlock()
|
||||||
|
|
||||||
@ -63,14 +63,9 @@ func (s *Span) End(opts ...trace.EndOption) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
var c trace.EndConfig
|
c := trace.SpanConfigure(opts)
|
||||||
for _, opt := range opts {
|
|
||||||
opt(&c)
|
|
||||||
}
|
|
||||||
|
|
||||||
s.endTime = time.Now()
|
s.endTime = time.Now()
|
||||||
|
if endTime := c.Timestamp; !endTime.IsZero() {
|
||||||
if endTime := c.EndTime; !endTime.IsZero() {
|
|
||||||
s.endTime = endTime
|
s.endTime = endTime
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -101,7 +101,7 @@ func TestSpan(t *testing.T) {
|
|||||||
e.Expect(endTime).ToEqual(expectedEndTime)
|
e.Expect(endTime).ToEqual(expectedEndTime)
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("uses the time from WithEndTime", func(t *testing.T) {
|
t.Run("uses the time from WithTimestamp", func(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
e := matchers.NewExpecter(t)
|
e := matchers.NewExpecter(t)
|
||||||
@ -113,7 +113,7 @@ func TestSpan(t *testing.T) {
|
|||||||
e.Expect(ok).ToBeTrue()
|
e.Expect(ok).ToBeTrue()
|
||||||
|
|
||||||
expectedEndTime := time.Now().AddDate(5, 0, 0)
|
expectedEndTime := time.Now().AddDate(5, 0, 0)
|
||||||
subject.End(trace.WithEndTime(expectedEndTime))
|
subject.End(trace.WithTimestamp(expectedEndTime))
|
||||||
|
|
||||||
e.Expect(subject.Ended()).ToBeTrue()
|
e.Expect(subject.Ended()).ToBeTrue()
|
||||||
|
|
||||||
|
@ -34,14 +34,10 @@ type Tracer struct {
|
|||||||
config *config
|
config *config
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Tracer) Start(ctx context.Context, name string, opts ...trace.StartOption) (context.Context, trace.Span) {
|
func (t *Tracer) Start(ctx context.Context, name string, opts ...trace.SpanOption) (context.Context, trace.Span) {
|
||||||
var c trace.StartConfig
|
c := trace.SpanConfigure(opts)
|
||||||
for _, opt := range opts {
|
|
||||||
opt(&c)
|
|
||||||
}
|
|
||||||
|
|
||||||
startTime := time.Now()
|
startTime := time.Now()
|
||||||
if st := c.StartTime; !st.IsZero() {
|
if st := c.Timestamp; !st.IsZero() {
|
||||||
startTime = st
|
startTime = st
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -47,7 +47,7 @@ func TestTracer(t *testing.T) {
|
|||||||
return span, nil
|
return span, nil
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("uses the start time from WithStartTime", func(t *testing.T) {
|
t.Run("uses the start time from WithTimestamp", func(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
e := matchers.NewExpecter(t)
|
e := matchers.NewExpecter(t)
|
||||||
@ -55,7 +55,7 @@ func TestTracer(t *testing.T) {
|
|||||||
expectedStartTime := time.Now().AddDate(5, 0, 0)
|
expectedStartTime := time.Now().AddDate(5, 0, 0)
|
||||||
|
|
||||||
subject := tp.Tracer(t.Name())
|
subject := tp.Tracer(t.Name())
|
||||||
_, span := subject.Start(context.Background(), "test", trace.WithStartTime(expectedStartTime))
|
_, span := subject.Start(context.Background(), "test", trace.WithTimestamp(expectedStartTime))
|
||||||
|
|
||||||
testSpan, ok := span.(*tracetest.Span)
|
testSpan, ok := span.(*tracetest.Span)
|
||||||
e.Expect(ok).ToBeTrue()
|
e.Expect(ok).ToBeTrue()
|
||||||
@ -223,7 +223,7 @@ func TestTracer(t *testing.T) {
|
|||||||
e.Expect(gotLinks).ToMatchInAnyOrder(expectedLinks)
|
e.Expect(gotLinks).ToMatchInAnyOrder(expectedLinks)
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("uses the links provided through LinkedTo", func(t *testing.T) {
|
t.Run("uses the links provided through WithLinks", func(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
e := matchers.NewExpecter(t)
|
e := matchers.NewExpecter(t)
|
||||||
@ -246,7 +246,7 @@ func TestTracer(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
_, span = subject.Start(context.Background(), "test", trace.LinkedTo(link1.SpanContext, link1.Attributes...), trace.LinkedTo(link2.SpanContext, link2.Attributes...))
|
_, span = subject.Start(context.Background(), "test", trace.WithLinks(link1, link2))
|
||||||
|
|
||||||
testSpan, ok := span.(*tracetest.Span)
|
testSpan, ok := span.(*tracetest.Span)
|
||||||
e.Expect(ok).ToBeTrue()
|
e.Expect(ok).ToBeTrue()
|
||||||
|
@ -99,10 +99,10 @@ func (s *bridgeSpan) Finish() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *bridgeSpan) FinishWithOptions(opts ot.FinishOptions) {
|
func (s *bridgeSpan) FinishWithOptions(opts ot.FinishOptions) {
|
||||||
var otelOpts []oteltrace.EndOption
|
var otelOpts []oteltrace.SpanOption
|
||||||
|
|
||||||
if !opts.FinishTime.IsZero() {
|
if !opts.FinishTime.IsZero() {
|
||||||
otelOpts = append(otelOpts, oteltrace.WithEndTime(opts.FinishTime))
|
otelOpts = append(otelOpts, oteltrace.WithTimestamp(opts.FinishTime))
|
||||||
}
|
}
|
||||||
for _, record := range opts.LogRecords {
|
for _, record := range opts.LogRecords {
|
||||||
s.logRecord(record)
|
s.logRecord(record)
|
||||||
@ -389,14 +389,15 @@ func (t *BridgeTracer) StartSpan(operationName string, opts ...ot.StartSpanOptio
|
|||||||
if parentBridgeSC != nil {
|
if parentBridgeSC != nil {
|
||||||
checkCtx = oteltrace.ContextWithRemoteSpanContext(checkCtx, parentBridgeSC.otelSpanContext)
|
checkCtx = oteltrace.ContextWithRemoteSpanContext(checkCtx, parentBridgeSC.otelSpanContext)
|
||||||
}
|
}
|
||||||
checkCtx2, otelSpan := t.setTracer.tracer().Start(checkCtx, operationName, func(opts *oteltrace.StartConfig) {
|
checkCtx2, otelSpan := t.setTracer.tracer().Start(
|
||||||
opts.Attributes = attributes
|
checkCtx,
|
||||||
opts.StartTime = sso.StartTime
|
operationName,
|
||||||
opts.Links = links
|
oteltrace.WithAttributes(attributes...),
|
||||||
opts.Record = true
|
oteltrace.WithTimestamp(sso.StartTime),
|
||||||
opts.NewRoot = false
|
oteltrace.WithLinks(links...),
|
||||||
opts.SpanKind = kind
|
oteltrace.WithRecord(),
|
||||||
})
|
oteltrace.WithSpanKind(kind),
|
||||||
|
)
|
||||||
if checkCtx != checkCtx2 {
|
if checkCtx != checkCtx2 {
|
||||||
t.warnOnce.Do(func() {
|
t.warnOnce.Do(func() {
|
||||||
t.warningHandler("SDK should have deferred the context setup, see the documentation of go.opentelemetry.io/otel/bridge/opentracing/migration\n")
|
t.warningHandler("SDK should have deferred the context setup, see the documentation of go.opentelemetry.io/otel/bridge/opentracing/migration\n")
|
||||||
|
@ -70,17 +70,14 @@ func NewMockTracer() *MockTracer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *MockTracer) Start(ctx context.Context, name string, opts ...oteltrace.StartOption) (context.Context, oteltrace.Span) {
|
func (t *MockTracer) Start(ctx context.Context, name string, opts ...oteltrace.SpanOption) (context.Context, oteltrace.Span) {
|
||||||
spanOpts := oteltrace.StartConfig{}
|
config := oteltrace.SpanConfigure(opts)
|
||||||
for _, opt := range opts {
|
startTime := config.Timestamp
|
||||||
opt(&spanOpts)
|
|
||||||
}
|
|
||||||
startTime := spanOpts.StartTime
|
|
||||||
if startTime.IsZero() {
|
if startTime.IsZero() {
|
||||||
startTime = time.Now()
|
startTime = time.Now()
|
||||||
}
|
}
|
||||||
spanContext := oteltrace.SpanContext{
|
spanContext := oteltrace.SpanContext{
|
||||||
TraceID: t.getTraceID(ctx, &spanOpts),
|
TraceID: t.getTraceID(ctx, config),
|
||||||
SpanID: t.getSpanID(),
|
SpanID: t.getSpanID(),
|
||||||
TraceFlags: 0,
|
TraceFlags: 0,
|
||||||
}
|
}
|
||||||
@ -88,15 +85,15 @@ func (t *MockTracer) Start(ctx context.Context, name string, opts ...oteltrace.S
|
|||||||
mockTracer: t,
|
mockTracer: t,
|
||||||
officialTracer: t,
|
officialTracer: t,
|
||||||
spanContext: spanContext,
|
spanContext: spanContext,
|
||||||
recording: spanOpts.Record,
|
recording: config.Record,
|
||||||
Attributes: otelcorrelation.NewMap(otelcorrelation.MapUpdate{
|
Attributes: otelcorrelation.NewMap(otelcorrelation.MapUpdate{
|
||||||
MultiKV: spanOpts.Attributes,
|
MultiKV: config.Attributes,
|
||||||
}),
|
}),
|
||||||
StartTime: startTime,
|
StartTime: startTime,
|
||||||
EndTime: time.Time{},
|
EndTime: time.Time{},
|
||||||
ParentSpanID: t.getParentSpanID(ctx, &spanOpts),
|
ParentSpanID: t.getParentSpanID(ctx, config),
|
||||||
Events: nil,
|
Events: nil,
|
||||||
SpanKind: oteltrace.ValidateSpanKind(spanOpts.SpanKind),
|
SpanKind: oteltrace.ValidateSpanKind(config.SpanKind),
|
||||||
}
|
}
|
||||||
if !migration.SkipContextSetup(ctx) {
|
if !migration.SkipContextSetup(ctx) {
|
||||||
ctx = oteltrace.ContextWithSpan(ctx, span)
|
ctx = oteltrace.ContextWithSpan(ctx, span)
|
||||||
@ -118,8 +115,8 @@ func (t *MockTracer) addSpareContextValue(ctx context.Context) context.Context {
|
|||||||
return ctx
|
return ctx
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *MockTracer) getTraceID(ctx context.Context, spanOpts *oteltrace.StartConfig) oteltrace.ID {
|
func (t *MockTracer) getTraceID(ctx context.Context, config *oteltrace.SpanConfig) oteltrace.ID {
|
||||||
if parent := t.getParentSpanContext(ctx, spanOpts); parent.IsValid() {
|
if parent := t.getParentSpanContext(ctx, config); parent.IsValid() {
|
||||||
return parent.TraceID
|
return parent.TraceID
|
||||||
}
|
}
|
||||||
if len(t.SpareTraceIDs) > 0 {
|
if len(t.SpareTraceIDs) > 0 {
|
||||||
@ -133,15 +130,15 @@ func (t *MockTracer) getTraceID(ctx context.Context, spanOpts *oteltrace.StartCo
|
|||||||
return t.getRandTraceID()
|
return t.getRandTraceID()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *MockTracer) getParentSpanID(ctx context.Context, spanOpts *oteltrace.StartConfig) oteltrace.SpanID {
|
func (t *MockTracer) getParentSpanID(ctx context.Context, config *oteltrace.SpanConfig) oteltrace.SpanID {
|
||||||
if parent := t.getParentSpanContext(ctx, spanOpts); parent.IsValid() {
|
if parent := t.getParentSpanContext(ctx, config); parent.IsValid() {
|
||||||
return parent.SpanID
|
return parent.SpanID
|
||||||
}
|
}
|
||||||
return oteltrace.SpanID{}
|
return oteltrace.SpanID{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *MockTracer) getParentSpanContext(ctx context.Context, spanOpts *oteltrace.StartConfig) oteltrace.SpanContext {
|
func (t *MockTracer) getParentSpanContext(ctx context.Context, config *oteltrace.SpanConfig) oteltrace.SpanContext {
|
||||||
spanCtx, _, _ := otelparent.GetSpanContextAndLinks(ctx, spanOpts.NewRoot)
|
spanCtx, _, _ := otelparent.GetSpanContextAndLinks(ctx, config.NewRoot)
|
||||||
return spanCtx
|
return spanCtx
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -239,17 +236,12 @@ func (s *MockSpan) applyUpdate(update otelcorrelation.MapUpdate) {
|
|||||||
s.Attributes = s.Attributes.Apply(update)
|
s.Attributes = s.Attributes.Apply(update)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *MockSpan) End(options ...oteltrace.EndOption) {
|
func (s *MockSpan) End(options ...oteltrace.SpanOption) {
|
||||||
if !s.EndTime.IsZero() {
|
if !s.EndTime.IsZero() {
|
||||||
return // already finished
|
return // already finished
|
||||||
}
|
}
|
||||||
endOpts := oteltrace.EndConfig{}
|
config := oteltrace.SpanConfigure(options)
|
||||||
|
endTime := config.Timestamp
|
||||||
for _, opt := range options {
|
|
||||||
opt(&endOpts)
|
|
||||||
}
|
|
||||||
|
|
||||||
endTime := endOpts.EndTime
|
|
||||||
if endTime.IsZero() {
|
if endTime.IsZero() {
|
||||||
endTime = time.Now()
|
endTime = time.Now()
|
||||||
}
|
}
|
||||||
|
@ -74,7 +74,7 @@ func (t *WrapperTracer) otelTracer() oteltrace.Tracer {
|
|||||||
// Start forwards the call to the wrapped tracer. It also tries to
|
// Start forwards the call to the wrapped tracer. It also tries to
|
||||||
// override the tracer of the returned span if the span implements the
|
// override the tracer of the returned span if the span implements the
|
||||||
// OverrideTracerSpanExtension interface.
|
// OverrideTracerSpanExtension interface.
|
||||||
func (t *WrapperTracer) Start(ctx context.Context, name string, opts ...oteltrace.StartOption) (context.Context, oteltrace.Span) {
|
func (t *WrapperTracer) Start(ctx context.Context, name string, opts ...oteltrace.SpanOption) (context.Context, oteltrace.Span) {
|
||||||
ctx, span := t.otelTracer().Start(ctx, name, opts...)
|
ctx, span := t.otelTracer().Start(ctx, name, opts...)
|
||||||
if spanWithExtension, ok := span.(migration.OverrideTracerSpanExtension); ok {
|
if spanWithExtension, ok := span.(migration.OverrideTracerSpanExtension); ok {
|
||||||
spanWithExtension.OverrideTracer(t)
|
spanWithExtension.OverrideTracer(t)
|
||||||
|
@ -108,8 +108,14 @@ func (s *span) SetAttribute(k string, v interface{}) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// End ends the span adding an error event if it was called while panicking.
|
// End ends the span.
|
||||||
func (s *span) End(options ...apitrace.EndOption) {
|
//
|
||||||
|
// The only SpanOption currently supported is WithTimestamp which will set the
|
||||||
|
// end time for a Span's life-cycle.
|
||||||
|
//
|
||||||
|
// If this method is called while panicking an error event is added to the
|
||||||
|
// Span before ending it and the panic is continued.
|
||||||
|
func (s *span) End(options ...apitrace.SpanOption) {
|
||||||
if s == nil {
|
if s == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -131,19 +137,16 @@ func (s *span) End(options ...apitrace.EndOption) {
|
|||||||
if !s.IsRecording() {
|
if !s.IsRecording() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
opts := apitrace.EndConfig{}
|
config := apitrace.SpanConfigure(options)
|
||||||
for _, opt := range options {
|
|
||||||
opt(&opts)
|
|
||||||
}
|
|
||||||
s.endOnce.Do(func() {
|
s.endOnce.Do(func() {
|
||||||
sps, _ := s.tracer.provider.spanProcessors.Load().(spanProcessorMap)
|
sps, _ := s.tracer.provider.spanProcessors.Load().(spanProcessorMap)
|
||||||
mustExportOrProcess := len(sps) > 0
|
mustExportOrProcess := len(sps) > 0
|
||||||
if mustExportOrProcess {
|
if mustExportOrProcess {
|
||||||
sd := s.makeSpanData()
|
sd := s.makeSpanData()
|
||||||
if opts.EndTime.IsZero() {
|
if config.Timestamp.IsZero() {
|
||||||
sd.EndTime = internal.MonotonicEndTime(sd.StartTime)
|
sd.EndTime = internal.MonotonicEndTime(sd.StartTime)
|
||||||
} else {
|
} else {
|
||||||
sd.EndTime = opts.EndTime
|
sd.EndTime = config.Timestamp
|
||||||
}
|
}
|
||||||
for sp := range sps {
|
for sp := range sps {
|
||||||
sp.OnEnd(sd)
|
sp.OnEnd(sd)
|
||||||
@ -324,7 +327,7 @@ func (s *span) addChild() {
|
|||||||
s.mu.Unlock()
|
s.mu.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func startSpanInternal(tr *tracer, name string, parent apitrace.SpanContext, remoteParent bool, o apitrace.StartConfig) *span {
|
func startSpanInternal(tr *tracer, name string, parent apitrace.SpanContext, remoteParent bool, o *apitrace.SpanConfig) *span {
|
||||||
var noParent bool
|
var noParent bool
|
||||||
span := &span{}
|
span := &span{}
|
||||||
span.spanContext = parent
|
span.spanContext = parent
|
||||||
@ -355,7 +358,7 @@ func startSpanInternal(tr *tracer, name string, parent apitrace.SpanContext, rem
|
|||||||
return span
|
return span
|
||||||
}
|
}
|
||||||
|
|
||||||
startTime := o.StartTime
|
startTime := o.Timestamp
|
||||||
if startTime.IsZero() {
|
if startTime.IsZero() {
|
||||||
startTime = time.Now()
|
startTime = time.Now()
|
||||||
}
|
}
|
||||||
|
@ -500,13 +500,11 @@ func TestLinks(t *testing.T) {
|
|||||||
sc1 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
sc1 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
||||||
sc2 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
sc2 := apitrace.SpanContext{TraceID: apitrace.ID([16]byte{1, 1}), SpanID: apitrace.SpanID{3}}
|
||||||
|
|
||||||
span := startSpan(tp, "Links",
|
links := []apitrace.Link{
|
||||||
apitrace.LinkedTo(sc1, label.String("key1", "value1")),
|
{SpanContext: sc1, Attributes: []label.KeyValue{k1v1}},
|
||||||
apitrace.LinkedTo(sc2,
|
{SpanContext: sc2, Attributes: []label.KeyValue{k2v2, k3v3}},
|
||||||
label.String("key2", "value2"),
|
}
|
||||||
label.String("key3", "value3"),
|
span := startSpan(tp, "Links", apitrace.WithLinks(links...))
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
got, err := endSpan(te, span)
|
got, err := endSpan(te, span)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -518,13 +516,10 @@ func TestLinks(t *testing.T) {
|
|||||||
TraceID: tid,
|
TraceID: tid,
|
||||||
TraceFlags: 0x1,
|
TraceFlags: 0x1,
|
||||||
},
|
},
|
||||||
ParentSpanID: sid,
|
ParentSpanID: sid,
|
||||||
Name: "span0",
|
Name: "span0",
|
||||||
HasRemoteParent: true,
|
HasRemoteParent: true,
|
||||||
Links: []apitrace.Link{
|
Links: links,
|
||||||
{SpanContext: sc1, Attributes: []label.KeyValue{k1v1}},
|
|
||||||
{SpanContext: sc2, Attributes: []label.KeyValue{k2v2, k3v3}},
|
|
||||||
},
|
|
||||||
SpanKind: apitrace.SpanKindInternal,
|
SpanKind: apitrace.SpanKindInternal,
|
||||||
InstrumentationLibrary: instrumentation.Library{Name: "Links"},
|
InstrumentationLibrary: instrumentation.Library{Name: "Links"},
|
||||||
}
|
}
|
||||||
@ -544,9 +539,11 @@ func TestLinksOverLimit(t *testing.T) {
|
|||||||
tp, _ := NewProvider(WithConfig(cfg), WithSyncer(te))
|
tp, _ := NewProvider(WithConfig(cfg), WithSyncer(te))
|
||||||
|
|
||||||
span := startSpan(tp, "LinksOverLimit",
|
span := startSpan(tp, "LinksOverLimit",
|
||||||
apitrace.LinkedTo(sc1, label.String("key1", "value1")),
|
apitrace.WithLinks(
|
||||||
apitrace.LinkedTo(sc2, label.String("key2", "value2")),
|
apitrace.Link{SpanContext: sc1, Attributes: []label.KeyValue{label.String("key1", "value1")}},
|
||||||
apitrace.LinkedTo(sc3, label.String("key3", "value3")),
|
apitrace.Link{SpanContext: sc2, Attributes: []label.KeyValue{label.String("key2", "value2")}},
|
||||||
|
apitrace.Link{SpanContext: sc3, Attributes: []label.KeyValue{label.String("key3", "value3")}},
|
||||||
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
k2v2 := label.String("key2", "value2")
|
k2v2 := label.String("key2", "value2")
|
||||||
@ -668,7 +665,7 @@ func checkChild(p apitrace.SpanContext, apiSpan apitrace.Span) error {
|
|||||||
|
|
||||||
// startSpan starts a span with a name "span0". See startNamedSpan for
|
// startSpan starts a span with a name "span0". See startNamedSpan for
|
||||||
// details.
|
// details.
|
||||||
func startSpan(tp *Provider, trName string, args ...apitrace.StartOption) apitrace.Span {
|
func startSpan(tp *Provider, trName string, args ...apitrace.SpanOption) apitrace.Span {
|
||||||
return startNamedSpan(tp, trName, "span0", args...)
|
return startNamedSpan(tp, trName, "span0", args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -676,7 +673,7 @@ func startSpan(tp *Provider, trName string, args ...apitrace.StartOption) apitra
|
|||||||
// passed name and with remote span context as parent. The remote span
|
// passed name and with remote span context as parent. The remote span
|
||||||
// context contains TraceFlags with sampled bit set. This allows the
|
// context contains TraceFlags with sampled bit set. This allows the
|
||||||
// span to be automatically sampled.
|
// span to be automatically sampled.
|
||||||
func startNamedSpan(tp *Provider, trName, name string, args ...apitrace.StartOption) apitrace.Span {
|
func startNamedSpan(tp *Provider, trName, name string, args ...apitrace.SpanOption) apitrace.Span {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
ctx = apitrace.ContextWithRemoteSpanContext(ctx, remoteSpanContext())
|
ctx = apitrace.ContextWithRemoteSpanContext(ctx, remoteSpanContext())
|
||||||
args = append(args, apitrace.WithRecord())
|
args = append(args, apitrace.WithRecord())
|
||||||
@ -906,9 +903,9 @@ func TestCustomStartEndTime(t *testing.T) {
|
|||||||
_, span := tp.Tracer("Custom Start and End time").Start(
|
_, span := tp.Tracer("Custom Start and End time").Start(
|
||||||
context.Background(),
|
context.Background(),
|
||||||
"testspan",
|
"testspan",
|
||||||
apitrace.WithStartTime(startTime),
|
apitrace.WithTimestamp(startTime),
|
||||||
)
|
)
|
||||||
span.End(apitrace.WithEndTime(endTime))
|
span.End(apitrace.WithTimestamp(endTime))
|
||||||
|
|
||||||
if len(te.spans) != 1 {
|
if len(te.spans) != 1 {
|
||||||
t.Fatalf("got exported spans %#v, want one span", te.spans)
|
t.Fatalf("got exported spans %#v, want one span", te.spans)
|
||||||
|
@ -29,14 +29,16 @@ type tracer struct {
|
|||||||
|
|
||||||
var _ apitrace.Tracer = &tracer{}
|
var _ apitrace.Tracer = &tracer{}
|
||||||
|
|
||||||
func (tr *tracer) Start(ctx context.Context, name string, o ...apitrace.StartOption) (context.Context, apitrace.Span) {
|
// Start starts a Span and returns it along with a context containing it.
|
||||||
var opts apitrace.StartConfig
|
//
|
||||||
|
// The Span is created with the provided name and as a child of any existing
|
||||||
|
// span context found in the passed context. The created Span will be
|
||||||
|
// configured appropriately by any SpanOption passed. Any Timestamp option
|
||||||
|
// passed will be used as the start time of the Span's life-cycle.
|
||||||
|
func (tr *tracer) Start(ctx context.Context, name string, options ...apitrace.SpanOption) (context.Context, apitrace.Span) {
|
||||||
|
config := apitrace.SpanConfigure(options)
|
||||||
|
|
||||||
for _, op := range o {
|
parentSpanContext, remoteParent, links := parent.GetSpanContextAndLinks(ctx, config.NewRoot)
|
||||||
op(&opts)
|
|
||||||
}
|
|
||||||
|
|
||||||
parentSpanContext, remoteParent, links := parent.GetSpanContextAndLinks(ctx, opts.NewRoot)
|
|
||||||
|
|
||||||
if p := apitrace.SpanFromContext(ctx); p != nil {
|
if p := apitrace.SpanFromContext(ctx); p != nil {
|
||||||
if sdkSpan, ok := p.(*span); ok {
|
if sdkSpan, ok := p.(*span); ok {
|
||||||
@ -44,14 +46,14 @@ func (tr *tracer) Start(ctx context.Context, name string, o ...apitrace.StartOpt
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
span := startSpanInternal(tr, name, parentSpanContext, remoteParent, opts)
|
span := startSpanInternal(tr, name, parentSpanContext, remoteParent, config)
|
||||||
for _, l := range links {
|
for _, l := range links {
|
||||||
span.addLink(l)
|
span.addLink(l)
|
||||||
}
|
}
|
||||||
for _, l := range opts.Links {
|
for _, l := range config.Links {
|
||||||
span.addLink(l)
|
span.addLink(l)
|
||||||
}
|
}
|
||||||
span.SetAttributes(opts.Attributes...)
|
span.SetAttributes(config.Attributes...)
|
||||||
|
|
||||||
span.tracer = tr
|
span.tracer = tr
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user