2023-05-09 20:19:48 +03:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2024-04-11 12:46:18 +03:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2023-05-09 20:19:48 +03:00
|
|
|
|
|
|
|
package resource // import "go.opentelemetry.io/otel/sdk/resource"
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
2024-03-11 17:34:34 +03:00
|
|
|
"fmt"
|
2023-05-09 20:19:48 +03:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"go.opentelemetry.io/otel"
|
|
|
|
"go.opentelemetry.io/otel/attribute"
|
2024-08-26 19:05:54 +03:00
|
|
|
"go.opentelemetry.io/otel/sdk/internal/x"
|
2023-05-09 20:19:48 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// Resource describes an entity about which identifying information
|
|
|
|
// and metadata is exposed. Resource is an immutable object,
|
|
|
|
// equivalent to a map from key to unique value.
|
|
|
|
//
|
|
|
|
// Resources should be passed and stored as pointers
|
|
|
|
// (`*resource.Resource`). The `nil` value is equivalent to an empty
|
|
|
|
// Resource.
|
|
|
|
type Resource struct {
|
|
|
|
attrs attribute.Set
|
|
|
|
schemaURL string
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
defaultResource *Resource
|
|
|
|
defaultResourceOnce sync.Once
|
|
|
|
)
|
|
|
|
|
2024-03-11 17:34:34 +03:00
|
|
|
// ErrSchemaURLConflict is an error returned when two Resources are merged
|
|
|
|
// together that contain different, non-empty, schema URLs.
|
|
|
|
var ErrSchemaURLConflict = errors.New("conflicting Schema URL")
|
2023-05-09 20:19:48 +03:00
|
|
|
|
2024-03-11 17:34:34 +03:00
|
|
|
// New returns a [Resource] built using opts.
|
|
|
|
//
|
|
|
|
// This may return a partial Resource along with an error containing
|
|
|
|
// [ErrPartialResource] if options that provide a [Detector] are used and that
|
|
|
|
// error is returned from one or more of the Detectors. It may also return a
|
|
|
|
// merge-conflict Resource along with an error containing
|
|
|
|
// [ErrSchemaURLConflict] if merging Resources from the opts results in a
|
|
|
|
// schema URL conflict (see [Resource.Merge] for more information). It is up to
|
|
|
|
// the caller to determine if this returned Resource should be used or not
|
|
|
|
// based on these errors.
|
2023-05-09 20:19:48 +03:00
|
|
|
func New(ctx context.Context, opts ...Option) (*Resource, error) {
|
|
|
|
cfg := config{}
|
|
|
|
for _, opt := range opts {
|
|
|
|
cfg = opt.apply(cfg)
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:15:05 +03:00
|
|
|
r := &Resource{schemaURL: cfg.schemaURL}
|
|
|
|
return r, detect(ctx, r, cfg.detectors)
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewWithAttributes creates a resource from attrs and associates the resource with a
|
|
|
|
// schema URL. If attrs contains duplicate keys, the last value will be used. If attrs
|
|
|
|
// contains any invalid items those items will be dropped. The attrs are assumed to be
|
|
|
|
// in a schema identified by schemaURL.
|
|
|
|
func NewWithAttributes(schemaURL string, attrs ...attribute.KeyValue) *Resource {
|
|
|
|
resource := NewSchemaless(attrs...)
|
|
|
|
resource.schemaURL = schemaURL
|
|
|
|
return resource
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSchemaless creates a resource from attrs. If attrs contains duplicate keys,
|
|
|
|
// the last value will be used. If attrs contains any invalid items those items will
|
|
|
|
// be dropped. The resource will not be associated with a schema URL. If the schema
|
|
|
|
// of the attrs is known use NewWithAttributes instead.
|
|
|
|
func NewSchemaless(attrs ...attribute.KeyValue) *Resource {
|
|
|
|
if len(attrs) == 0 {
|
2023-09-07 14:20:37 +03:00
|
|
|
return &Resource{}
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure attributes comply with the specification:
|
2023-06-05 11:15:05 +03:00
|
|
|
// https://github.com/open-telemetry/opentelemetry-specification/blob/v1.20.0/specification/common/README.md#attribute
|
2023-05-09 20:19:48 +03:00
|
|
|
s, _ := attribute.NewSetWithFiltered(attrs, func(kv attribute.KeyValue) bool {
|
|
|
|
return kv.Valid()
|
|
|
|
})
|
|
|
|
|
|
|
|
// If attrs only contains invalid entries do not allocate a new resource.
|
|
|
|
if s.Len() == 0 {
|
2023-09-07 14:20:37 +03:00
|
|
|
return &Resource{}
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return &Resource{attrs: s} //nolint
|
|
|
|
}
|
|
|
|
|
|
|
|
// String implements the Stringer interface and provides a
|
|
|
|
// human-readable form of the resource.
|
|
|
|
//
|
|
|
|
// Avoid using this representation as the key in a map of resources,
|
|
|
|
// use Equivalent() as the key instead.
|
|
|
|
func (r *Resource) String() string {
|
|
|
|
if r == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return r.attrs.Encoded(attribute.DefaultEncoder())
|
|
|
|
}
|
|
|
|
|
2024-03-11 17:34:34 +03:00
|
|
|
// MarshalLog is the marshaling function used by the logging system to represent this Resource.
|
2023-05-09 20:19:48 +03:00
|
|
|
func (r *Resource) MarshalLog() interface{} {
|
|
|
|
return struct {
|
|
|
|
Attributes attribute.Set
|
|
|
|
SchemaURL string
|
|
|
|
}{
|
|
|
|
Attributes: r.attrs,
|
|
|
|
SchemaURL: r.schemaURL,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attributes returns a copy of attributes from the resource in a sorted order.
|
|
|
|
// To avoid allocating a new slice, use an iterator.
|
|
|
|
func (r *Resource) Attributes() []attribute.KeyValue {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return r.attrs.ToSlice()
|
|
|
|
}
|
|
|
|
|
|
|
|
// SchemaURL returns the schema URL associated with Resource r.
|
|
|
|
func (r *Resource) SchemaURL() string {
|
|
|
|
if r == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return r.schemaURL
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iter returns an iterator of the Resource attributes.
|
|
|
|
// This is ideal to use if you do not want a copy of the attributes.
|
|
|
|
func (r *Resource) Iter() attribute.Iterator {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return r.attrs.Iter()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Equal returns true when a Resource is equivalent to this Resource.
|
|
|
|
func (r *Resource) Equal(eq *Resource) bool {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
if eq == nil {
|
|
|
|
eq = Empty()
|
|
|
|
}
|
|
|
|
return r.Equivalent() == eq.Equivalent()
|
|
|
|
}
|
|
|
|
|
2024-03-11 17:34:34 +03:00
|
|
|
// Merge creates a new [Resource] by merging a and b.
|
|
|
|
//
|
|
|
|
// If there are common keys between a and b, then the value from b will
|
|
|
|
// overwrite the value from a, even if b's value is empty.
|
2023-05-09 20:19:48 +03:00
|
|
|
//
|
2024-03-11 17:34:34 +03:00
|
|
|
// The SchemaURL of the resources will be merged according to the
|
|
|
|
// [OpenTelemetry specification rules]:
|
2023-05-09 20:19:48 +03:00
|
|
|
//
|
2024-03-11 17:34:34 +03:00
|
|
|
// - If a's schema URL is empty then the returned Resource's schema URL will
|
|
|
|
// be set to the schema URL of b,
|
|
|
|
// - Else if b's schema URL is empty then the returned Resource's schema URL
|
|
|
|
// will be set to the schema URL of a,
|
|
|
|
// - Else if the schema URLs of a and b are the same then that will be the
|
|
|
|
// schema URL of the returned Resource,
|
|
|
|
// - Else this is a merging error. If the resources have different,
|
|
|
|
// non-empty, schema URLs an error containing [ErrSchemaURLConflict] will
|
|
|
|
// be returned with the merged Resource. The merged Resource will have an
|
|
|
|
// empty schema URL. It may be the case that some unintended attributes
|
|
|
|
// have been overwritten or old semantic conventions persisted in the
|
|
|
|
// returned Resource. It is up to the caller to determine if this returned
|
|
|
|
// Resource should be used or not.
|
|
|
|
//
|
|
|
|
// [OpenTelemetry specification rules]: https://github.com/open-telemetry/opentelemetry-specification/blob/v1.20.0/specification/resource/sdk.md#merge
|
2023-05-09 20:19:48 +03:00
|
|
|
func Merge(a, b *Resource) (*Resource, error) {
|
|
|
|
if a == nil && b == nil {
|
|
|
|
return Empty(), nil
|
|
|
|
}
|
|
|
|
if a == nil {
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
if b == nil {
|
|
|
|
return a, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note: 'b' attributes will overwrite 'a' with last-value-wins in attribute.Key()
|
|
|
|
// Meaning this is equivalent to: append(a.Attributes(), b.Attributes()...)
|
|
|
|
mi := attribute.NewMergeIterator(b.Set(), a.Set())
|
|
|
|
combine := make([]attribute.KeyValue, 0, a.Len()+b.Len())
|
|
|
|
for mi.Next() {
|
|
|
|
combine = append(combine, mi.Attribute())
|
|
|
|
}
|
2024-03-11 17:34:34 +03:00
|
|
|
|
|
|
|
switch {
|
|
|
|
case a.schemaURL == "":
|
|
|
|
return NewWithAttributes(b.schemaURL, combine...), nil
|
|
|
|
case b.schemaURL == "":
|
|
|
|
return NewWithAttributes(a.schemaURL, combine...), nil
|
|
|
|
case a.schemaURL == b.schemaURL:
|
|
|
|
return NewWithAttributes(a.schemaURL, combine...), nil
|
|
|
|
}
|
|
|
|
// Return the merged resource with an appropriate error. It is up to
|
|
|
|
// the user to decide if the returned resource can be used or not.
|
|
|
|
return NewSchemaless(combine...), fmt.Errorf(
|
|
|
|
"%w: %s and %s",
|
|
|
|
ErrSchemaURLConflict,
|
|
|
|
a.schemaURL,
|
|
|
|
b.schemaURL,
|
|
|
|
)
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Empty returns an instance of Resource with no attributes. It is
|
|
|
|
// equivalent to a `nil` Resource.
|
|
|
|
func Empty() *Resource {
|
2023-09-07 14:20:37 +03:00
|
|
|
return &Resource{}
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Default returns an instance of Resource with a default
|
|
|
|
// "service.name" and OpenTelemetrySDK attributes.
|
|
|
|
func Default() *Resource {
|
|
|
|
defaultResourceOnce.Do(func() {
|
|
|
|
var err error
|
2024-08-26 19:05:54 +03:00
|
|
|
defaultDetectors := []Detector{
|
2023-05-09 20:19:48 +03:00
|
|
|
defaultServiceNameDetector{},
|
|
|
|
fromEnv{},
|
|
|
|
telemetrySDK{},
|
2024-08-26 19:05:54 +03:00
|
|
|
}
|
|
|
|
if x.Resource.Enabled() {
|
|
|
|
defaultDetectors = append([]Detector{defaultServiceInstanceIDDetector{}}, defaultDetectors...)
|
|
|
|
}
|
|
|
|
defaultResource, err = Detect(
|
|
|
|
context.Background(),
|
|
|
|
defaultDetectors...,
|
2023-05-09 20:19:48 +03:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
otel.Handle(err)
|
|
|
|
}
|
|
|
|
// If Detect did not return a valid resource, fall back to emptyResource.
|
|
|
|
if defaultResource == nil {
|
2023-09-07 14:20:37 +03:00
|
|
|
defaultResource = &Resource{}
|
2023-05-09 20:19:48 +03:00
|
|
|
}
|
|
|
|
})
|
|
|
|
return defaultResource
|
|
|
|
}
|
|
|
|
|
|
|
|
// Environment returns an instance of Resource with attributes
|
|
|
|
// extracted from the OTEL_RESOURCE_ATTRIBUTES environment variable.
|
|
|
|
func Environment() *Resource {
|
|
|
|
detector := &fromEnv{}
|
|
|
|
resource, err := detector.Detect(context.Background())
|
|
|
|
if err != nil {
|
|
|
|
otel.Handle(err)
|
|
|
|
}
|
|
|
|
return resource
|
|
|
|
}
|
|
|
|
|
|
|
|
// Equivalent returns an object that can be compared for equality
|
|
|
|
// between two resources. This value is suitable for use as a key in
|
|
|
|
// a map.
|
|
|
|
func (r *Resource) Equivalent() attribute.Distinct {
|
|
|
|
return r.Set().Equivalent()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set returns the equivalent *attribute.Set of this resource's attributes.
|
|
|
|
func (r *Resource) Set() *attribute.Set {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return &r.attrs
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON encodes the resource attributes as a JSON list of { "Key":
|
|
|
|
// "...", "Value": ... } pairs in order sorted by key.
|
|
|
|
func (r *Resource) MarshalJSON() ([]byte, error) {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return r.attrs.MarshalJSON()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Len returns the number of unique key-values in this Resource.
|
|
|
|
func (r *Resource) Len() int {
|
|
|
|
if r == nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return r.attrs.Len()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encoded returns an encoded representation of the resource.
|
|
|
|
func (r *Resource) Encoded(enc attribute.Encoder) string {
|
|
|
|
if r == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return r.attrs.Encoded(enc)
|
|
|
|
}
|