mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2024-12-16 10:19:23 +02:00
961121698b
* Move scope.Active to trace.CurrentSpan * Remove scope / does not build * Global tracer * Checkpoint * Checkpoint * Add key/key.go for key.New * Comments * Remove more EventID and ScopeID * Use Handle to describe static objects * TODOs * Remove empty file * Remove singletons * Update TODOs * TODO about map update * Make stats package option aliases (like key has) * Rename experimental/streaming * streaming SDK builds w/ many TODOs * Get the examples building * Tidy up metric API / add interface check * Remove logic from the registry; this is now a placeholder
199 lines
4.1 KiB
Go
199 lines
4.1 KiB
Go
// 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.
|
|
|
|
package trace
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"github.com/open-telemetry/opentelemetry-go/api/core"
|
|
"github.com/open-telemetry/opentelemetry-go/api/event"
|
|
"github.com/open-telemetry/opentelemetry-go/api/tag"
|
|
apitrace "github.com/open-telemetry/opentelemetry-go/api/trace"
|
|
"github.com/open-telemetry/opentelemetry-go/experimental/streaming/exporter/observer"
|
|
)
|
|
|
|
type span struct {
|
|
tracer *tracer
|
|
spanContext core.SpanContext
|
|
lock sync.Mutex
|
|
eventID observer.EventID
|
|
finishOnce sync.Once
|
|
recordEvent bool
|
|
status codes.Code
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
func (sp *span) ScopeID() observer.ScopeID {
|
|
if sp == nil {
|
|
return observer.ScopeID{}
|
|
}
|
|
sp.lock.Lock()
|
|
sid := observer.ScopeID{
|
|
EventID: sp.eventID,
|
|
SpanContext: sp.spanContext,
|
|
}
|
|
sp.lock.Unlock()
|
|
return sid
|
|
}
|
|
|
|
func (sp *span) updateScope() (observer.ScopeID, observer.EventID) {
|
|
next := observer.NextEventID()
|
|
|
|
sp.lock.Lock()
|
|
sid := observer.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 tag.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 ...tag.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)
|
|
}
|
|
}
|
|
|
|
func (sp *span) Tracer() apitrace.Tracer {
|
|
return sp.tracer
|
|
}
|
|
|
|
func (sp *span) AddEvent(ctx context.Context, event event.Event) {
|
|
observer.Record(observer.Event{
|
|
Type: observer.ADD_EVENT,
|
|
String: event.Message(),
|
|
Attributes: event.Attributes(),
|
|
Context: ctx,
|
|
})
|
|
}
|
|
|
|
func (sp *span) Event(ctx context.Context, msg string, attrs ...core.KeyValue) {
|
|
observer.Record(observer.Event{
|
|
Type: observer.ADD_EVENT,
|
|
String: msg,
|
|
Attributes: attrs,
|
|
Context: ctx,
|
|
})
|
|
}
|