2020-03-25 14:47:17 -07:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2020-03-09 09:30:42 -07:00
|
|
|
//
|
|
|
|
// 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 resource
|
|
|
|
|
|
|
|
import (
|
2020-05-14 07:06:03 +08:00
|
|
|
"go.opentelemetry.io/otel/api/kv"
|
2020-04-23 12:10:58 -07:00
|
|
|
"go.opentelemetry.io/otel/api/label"
|
2020-03-09 09:30:42 -07:00
|
|
|
)
|
|
|
|
|
2020-04-23 12:10:58 -07: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.
|
2020-03-09 09:30:42 -07:00
|
|
|
type Resource struct {
|
2020-04-23 12:10:58 -07:00
|
|
|
labels label.Set
|
2020-03-09 09:30:42 -07:00
|
|
|
}
|
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
var emptyResource Resource
|
|
|
|
|
2020-07-20 11:43:51 -04:00
|
|
|
// New creates a resource from a set of attributes. If there are
|
2020-04-23 12:10:58 -07:00
|
|
|
// duplicate keys present in the list of attributes, then the last
|
|
|
|
// value found for the key is preserved.
|
2020-05-14 07:06:03 +08:00
|
|
|
func New(kvs ...kv.KeyValue) *Resource {
|
2020-04-23 12:10:58 -07:00
|
|
|
return &Resource{
|
|
|
|
labels: label.NewSet(kvs...),
|
2020-03-09 09:30:42 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
// 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 ""
|
2020-04-07 12:15:36 -07:00
|
|
|
}
|
2020-04-23 12:10:58 -07:00
|
|
|
return r.labels.Encoded(label.DefaultEncoder())
|
2020-04-07 12:15:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Attributes returns a copy of attributes from the resource in a sorted order.
|
2020-04-23 12:10:58 -07:00
|
|
|
// To avoid allocating a new slice, use an iterator.
|
2020-05-14 07:06:03 +08:00
|
|
|
func (r *Resource) Attributes() []kv.KeyValue {
|
2020-04-23 12:10:58 -07:00
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return r.labels.ToSlice()
|
2020-04-07 12:15:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Iter returns an interator of the Resource attributes.
|
|
|
|
// This is ideal to use if you do not want a copy of the attributes.
|
2020-04-23 12:10:58 -07:00
|
|
|
func (r *Resource) Iter() label.Iterator {
|
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
|
|
|
return r.labels.Iter()
|
2020-04-07 12:15:36 -07:00
|
|
|
}
|
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
// 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()
|
2020-04-07 12:15:36 -07:00
|
|
|
}
|
|
|
|
|
2020-03-09 09:30:42 -07:00
|
|
|
// Merge creates a new resource by combining resource a and b.
|
2020-04-23 12:10:58 -07:00
|
|
|
//
|
|
|
|
// If there are common keys between resource a and b, then the value
|
|
|
|
// from resource a is preserved.
|
2020-03-09 09:30:42 -07:00
|
|
|
func Merge(a, b *Resource) *Resource {
|
2020-05-21 00:33:39 -07:00
|
|
|
if a == nil && b == nil {
|
|
|
|
return Empty()
|
|
|
|
}
|
2020-03-09 09:30:42 -07:00
|
|
|
if a == nil {
|
2020-05-21 00:33:39 -07:00
|
|
|
return b
|
2020-03-09 09:30:42 -07:00
|
|
|
}
|
|
|
|
if b == nil {
|
2020-05-21 00:33:39 -07:00
|
|
|
return a
|
2020-03-09 09:30:42 -07:00
|
|
|
}
|
2020-05-21 00:33:39 -07:00
|
|
|
|
2020-05-21 09:53:34 -07:00
|
|
|
// Note: 'a' labels will overwrite 'b' with last-value-wins in label.Key()
|
|
|
|
// Meaning this is equivalent to: append(b.Attributes(), a.Attributes()...)
|
2020-05-21 00:33:39 -07:00
|
|
|
mi := label.NewMergeIterator(a.LabelSet(), b.LabelSet())
|
|
|
|
combine := make([]kv.KeyValue, 0, a.Len()+b.Len())
|
|
|
|
for mi.Next() {
|
|
|
|
combine = append(combine, mi.Label())
|
|
|
|
}
|
2020-04-23 12:10:58 -07:00
|
|
|
return New(combine...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Empty returns an instance of Resource with no attributes. It is
|
|
|
|
// equivalent to a `nil` Resource.
|
|
|
|
func Empty() *Resource {
|
|
|
|
return &emptyResource
|
|
|
|
}
|
2020-03-09 09:30:42 -07:00
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
// 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() label.Distinct {
|
2020-05-21 00:33:39 -07:00
|
|
|
return r.LabelSet().Equivalent()
|
|
|
|
}
|
|
|
|
|
|
|
|
// LabelSet returns the equivalent *label.Set.
|
|
|
|
func (r *Resource) LabelSet() *label.Set {
|
2020-04-23 12:10:58 -07:00
|
|
|
if r == nil {
|
|
|
|
r = Empty()
|
|
|
|
}
|
2020-05-21 00:33:39 -07:00
|
|
|
return &r.labels
|
2020-04-23 12:10:58 -07:00
|
|
|
}
|
2020-03-20 08:58:32 -07:00
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
// MarshalJSON encodes labels 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.labels.MarshalJSON()
|
2020-03-20 08:58:32 -07:00
|
|
|
}
|
2020-04-22 14:32:58 -07:00
|
|
|
|
2020-04-23 12:10:58 -07:00
|
|
|
// Len returns the number of unique key-values in this Resource.
|
|
|
|
func (r *Resource) Len() int {
|
|
|
|
if r == nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return r.labels.Len()
|
2020-04-22 14:32:58 -07:00
|
|
|
}
|
2020-04-24 09:44:21 -07:00
|
|
|
|
|
|
|
// Encoded returns an encoded representation of the resource by
|
|
|
|
// applying a label encoder. The result is cached by the underlying
|
|
|
|
// label set.
|
|
|
|
func (r *Resource) Encoded(enc label.Encoder) string {
|
|
|
|
if r == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return r.labels.Encoded(enc)
|
|
|
|
}
|