2019-06-14 22:09:41 +02:00
|
|
|
// Copyright 2019, 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.
|
|
|
|
|
2019-06-14 20:37:05 +02:00
|
|
|
package trace
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-07-01 20:23:26 +02:00
|
|
|
"sync"
|
2019-06-14 20:37:05 +02:00
|
|
|
|
2019-06-19 19:44:46 +02:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
|
2019-06-14 20:37:05 +02:00
|
|
|
"github.com/open-telemetry/opentelemetry-go/api/core"
|
2019-06-25 22:50:46 +02:00
|
|
|
"github.com/open-telemetry/opentelemetry-go/api/event"
|
2019-06-14 20:37:05 +02:00
|
|
|
"github.com/open-telemetry/opentelemetry-go/api/stats"
|
2019-06-27 07:03:09 +02:00
|
|
|
apitrace "github.com/open-telemetry/opentelemetry-go/api/trace"
|
2019-06-14 20:37:05 +02:00
|
|
|
"github.com/open-telemetry/opentelemetry-go/exporter/observer"
|
|
|
|
)
|
|
|
|
|
2019-06-27 22:34:57 +02:00
|
|
|
type span struct {
|
|
|
|
tracer *tracer
|
|
|
|
spanContext core.SpanContext
|
|
|
|
lock sync.Mutex
|
|
|
|
eventID core.EventID
|
|
|
|
finishOnce sync.Once
|
|
|
|
recordEvent bool
|
|
|
|
status codes.Code
|
|
|
|
}
|
|
|
|
|
2019-06-19 19:44:46 +02:00
|
|
|
// SpancContext returns span context of the span. Return SpanContext is usable
|
|
|
|
// even after the span is finished.
|
|
|
|
func (sp *span) SpanContext() core.SpanContext {
|
|
|
|
if sp == nil {
|
|
|
|
return core.INVALID_SPAN_CONTEXT
|
|
|
|
}
|
|
|
|
return sp.spanContext
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsRecordingEvents returns true is the span is active and recording events is enabled.
|
|
|
|
func (sp *span) IsRecordingEvents() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetStatus sets the status of the span.
|
|
|
|
func (sp *span) SetStatus(status codes.Code) {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
sid := sp.ScopeID()
|
|
|
|
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.SET_STATUS,
|
|
|
|
Scope: sid,
|
|
|
|
Sequence: sid.EventID,
|
|
|
|
Status: status,
|
|
|
|
})
|
|
|
|
sp.status = status
|
|
|
|
}
|
|
|
|
|
2019-06-14 20:37:05 +02:00
|
|
|
func (sp *span) ScopeID() core.ScopeID {
|
|
|
|
if sp == nil {
|
|
|
|
return core.ScopeID{}
|
|
|
|
}
|
|
|
|
sp.lock.Lock()
|
|
|
|
sid := core.ScopeID{
|
|
|
|
EventID: sp.eventID,
|
|
|
|
SpanContext: sp.spanContext,
|
|
|
|
}
|
|
|
|
sp.lock.Unlock()
|
|
|
|
return sid
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) updateScope() (core.ScopeID, core.EventID) {
|
|
|
|
next := observer.NextEventID()
|
|
|
|
|
|
|
|
sp.lock.Lock()
|
|
|
|
sid := core.ScopeID{
|
|
|
|
EventID: sp.eventID,
|
|
|
|
SpanContext: sp.spanContext,
|
|
|
|
}
|
|
|
|
sp.eventID = next
|
|
|
|
sp.lock.Unlock()
|
|
|
|
|
|
|
|
return sid, next
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) SetError(v bool) {
|
|
|
|
sp.SetAttribute(ErrorKey.Bool(v))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) SetAttribute(attribute core.KeyValue) {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
sid, next := sp.updateScope()
|
|
|
|
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.MODIFY_ATTR,
|
|
|
|
Scope: sid,
|
|
|
|
Sequence: next,
|
|
|
|
Attribute: attribute,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) SetAttributes(attributes ...core.KeyValue) {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
sid, next := sp.updateScope()
|
|
|
|
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.MODIFY_ATTR,
|
|
|
|
Scope: sid,
|
|
|
|
Sequence: next,
|
|
|
|
Attributes: attributes,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) ModifyAttribute(mutator core.Mutator) {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
sid, next := sp.updateScope()
|
|
|
|
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.MODIFY_ATTR,
|
|
|
|
Scope: sid,
|
|
|
|
Sequence: next,
|
|
|
|
Mutator: mutator,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) ModifyAttributes(mutators ...core.Mutator) {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
sid, next := sp.updateScope()
|
|
|
|
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.MODIFY_ATTR,
|
|
|
|
Scope: sid,
|
|
|
|
Sequence: next,
|
|
|
|
Mutators: mutators,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) Finish() {
|
|
|
|
if sp == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
recovered := recover()
|
|
|
|
sp.finishOnce.Do(func() {
|
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.FINISH_SPAN,
|
|
|
|
Scope: sp.ScopeID(),
|
|
|
|
Recovered: recovered,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
if recovered != nil {
|
|
|
|
panic(recovered)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-27 07:03:09 +02:00
|
|
|
func (sp *span) Tracer() apitrace.Tracer {
|
2019-06-14 20:37:05 +02:00
|
|
|
return sp.tracer
|
|
|
|
}
|
|
|
|
|
2019-06-25 22:50:46 +02:00
|
|
|
func (sp *span) AddEvent(ctx context.Context, event event.Event) {
|
2019-06-24 19:35:16 +02:00
|
|
|
|
2019-06-25 22:50:46 +02:00
|
|
|
observer.Record(observer.Event{
|
|
|
|
Type: observer.ADD_EVENT,
|
|
|
|
Event: event,
|
|
|
|
Context: ctx,
|
|
|
|
})
|
2019-06-24 19:35:16 +02:00
|
|
|
}
|
|
|
|
|
2019-06-14 20:37:05 +02:00
|
|
|
func (sp *span) Record(ctx context.Context, m ...core.Measurement) {
|
|
|
|
stats.With(sp).Record(ctx, m...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *span) RecordSingle(ctx context.Context, m core.Measurement) {
|
|
|
|
stats.With(sp).RecordSingle(ctx, m)
|
|
|
|
}
|