Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
300 lines
7.2 KiB
300 lines
7.2 KiB
// 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 label // import "go.opentelemetry.io/otel/label" |
|
|
|
import ( |
|
"encoding/json" |
|
"fmt" |
|
"reflect" |
|
"strconv" |
|
"unsafe" |
|
|
|
"go.opentelemetry.io/otel/internal" |
|
) |
|
|
|
//go:generate stringer -type=Type |
|
|
|
// Type describes the type of the data Value holds. |
|
type Type int |
|
|
|
// Value represents the value part in key-value pairs. |
|
type Value struct { |
|
vtype Type |
|
numeric uint64 |
|
stringly string |
|
// TODO Lazy value type? |
|
|
|
array interface{} |
|
} |
|
|
|
const ( |
|
// INVALID is used for a Value with no value set. |
|
INVALID Type = iota |
|
// BOOL is a boolean Type Value. |
|
BOOL |
|
// INT32 is a 32-bit signed integral Type Value. |
|
INT32 |
|
// INT64 is a 64-bit signed integral Type Value. |
|
INT64 |
|
// UINT32 is a 32-bit unsigned integral Type Value. |
|
UINT32 |
|
// UINT64 is a 64-bit unsigned integral Type Value. |
|
UINT64 |
|
// FLOAT32 is a 32-bit floating point Type Value. |
|
FLOAT32 |
|
// FLOAT64 is a 64-bit floating point Type Value. |
|
FLOAT64 |
|
// STRING is a string Type Value. |
|
STRING |
|
// ARRAY is an array Type Value used to store 1-dimensional slices or |
|
// arrays of bool, int, int32, int64, uint, uint32, uint64, float, |
|
// float32, float64, or string types. |
|
ARRAY |
|
) |
|
|
|
// BoolValue creates a BOOL Value. |
|
func BoolValue(v bool) Value { |
|
return Value{ |
|
vtype: BOOL, |
|
numeric: internal.BoolToRaw(v), |
|
} |
|
} |
|
|
|
// Int64Value creates an INT64 Value. |
|
func Int64Value(v int64) Value { |
|
return Value{ |
|
vtype: INT64, |
|
numeric: internal.Int64ToRaw(v), |
|
} |
|
} |
|
|
|
// Uint64Value creates a UINT64 Value. |
|
func Uint64Value(v uint64) Value { |
|
return Value{ |
|
vtype: UINT64, |
|
numeric: internal.Uint64ToRaw(v), |
|
} |
|
} |
|
|
|
// Float64Value creates a FLOAT64 Value. |
|
func Float64Value(v float64) Value { |
|
return Value{ |
|
vtype: FLOAT64, |
|
numeric: internal.Float64ToRaw(v), |
|
} |
|
} |
|
|
|
// Int32Value creates an INT32 Value. |
|
func Int32Value(v int32) Value { |
|
return Value{ |
|
vtype: INT32, |
|
numeric: internal.Int32ToRaw(v), |
|
} |
|
} |
|
|
|
// Uint32Value creates a UINT32 Value. |
|
func Uint32Value(v uint32) Value { |
|
return Value{ |
|
vtype: UINT32, |
|
numeric: internal.Uint32ToRaw(v), |
|
} |
|
} |
|
|
|
// Float32Value creates a FLOAT32 Value. |
|
func Float32Value(v float32) Value { |
|
return Value{ |
|
vtype: FLOAT32, |
|
numeric: internal.Float32ToRaw(v), |
|
} |
|
} |
|
|
|
// StringValue creates a STRING Value. |
|
func StringValue(v string) Value { |
|
return Value{ |
|
vtype: STRING, |
|
stringly: v, |
|
} |
|
} |
|
|
|
// IntValue creates either an INT32 or an INT64 Value, depending on whether |
|
// the int type is 32 or 64 bits wide. |
|
func IntValue(v int) Value { |
|
if unsafe.Sizeof(v) == 4 { |
|
return Int32Value(int32(v)) |
|
} |
|
return Int64Value(int64(v)) |
|
} |
|
|
|
// UintValue creates either a UINT32 or a UINT64 Value, depending on whether |
|
// the uint type is 32 or 64 bits wide. |
|
func UintValue(v uint) Value { |
|
if unsafe.Sizeof(v) == 4 { |
|
return Uint32Value(uint32(v)) |
|
} |
|
return Uint64Value(uint64(v)) |
|
} |
|
|
|
// ArrayValue creates an ARRAY value from an array or slice. |
|
// Only arrays or slices of bool, int, int32, int64, uint, uint32, uint64, |
|
// float, float32, float64, or string types are allowed. Specifically, arrays |
|
// and slices can not contain other arrays, slices, structs, or non-standard |
|
// types. If the passed value is not an array or slice of these types an |
|
// INVALID value is returned. |
|
func ArrayValue(v interface{}) Value { |
|
switch reflect.TypeOf(v).Kind() { |
|
case reflect.Array, reflect.Slice: |
|
// get array type regardless of dimensions |
|
typ := reflect.TypeOf(v).Elem() |
|
kind := typ.Kind() |
|
switch kind { |
|
case reflect.Bool, reflect.Int, reflect.Int32, reflect.Int64, |
|
reflect.Float32, reflect.Float64, reflect.String, |
|
reflect.Uint, reflect.Uint32, reflect.Uint64: |
|
val := reflect.ValueOf(v) |
|
length := val.Len() |
|
frozen := reflect.Indirect(reflect.New(reflect.ArrayOf(length, typ))) |
|
reflect.Copy(frozen, val) |
|
return Value{ |
|
vtype: ARRAY, |
|
array: frozen.Interface(), |
|
} |
|
default: |
|
return Value{vtype: INVALID} |
|
} |
|
} |
|
return Value{vtype: INVALID} |
|
} |
|
|
|
// Type returns a type of the Value. |
|
func (v Value) Type() Type { |
|
return v.vtype |
|
} |
|
|
|
// AsBool returns the bool value. Make sure that the Value's type is |
|
// BOOL. |
|
func (v Value) AsBool() bool { |
|
return internal.RawToBool(v.numeric) |
|
} |
|
|
|
// AsInt32 returns the int32 value. Make sure that the Value's type is |
|
// INT32. |
|
func (v Value) AsInt32() int32 { |
|
return internal.RawToInt32(v.numeric) |
|
} |
|
|
|
// AsInt64 returns the int64 value. Make sure that the Value's type is |
|
// INT64. |
|
func (v Value) AsInt64() int64 { |
|
return internal.RawToInt64(v.numeric) |
|
} |
|
|
|
// AsUint32 returns the uint32 value. Make sure that the Value's type |
|
// is UINT32. |
|
func (v Value) AsUint32() uint32 { |
|
return internal.RawToUint32(v.numeric) |
|
} |
|
|
|
// AsUint64 returns the uint64 value. Make sure that the Value's type is |
|
// UINT64. |
|
func (v Value) AsUint64() uint64 { |
|
return internal.RawToUint64(v.numeric) |
|
} |
|
|
|
// AsFloat32 returns the float32 value. Make sure that the Value's |
|
// type is FLOAT32. |
|
func (v Value) AsFloat32() float32 { |
|
return internal.RawToFloat32(v.numeric) |
|
} |
|
|
|
// AsFloat64 returns the float64 value. Make sure that the Value's |
|
// type is FLOAT64. |
|
func (v Value) AsFloat64() float64 { |
|
return internal.RawToFloat64(v.numeric) |
|
} |
|
|
|
// AsString returns the string value. Make sure that the Value's type |
|
// is STRING. |
|
func (v Value) AsString() string { |
|
return v.stringly |
|
} |
|
|
|
// AsArray returns the array Value as an interface{}. |
|
func (v Value) AsArray() interface{} { |
|
return v.array |
|
} |
|
|
|
type unknownValueType struct{} |
|
|
|
// AsInterface returns Value's data as interface{}. |
|
func (v Value) AsInterface() interface{} { |
|
switch v.Type() { |
|
case ARRAY: |
|
return v.AsArray() |
|
case BOOL: |
|
return v.AsBool() |
|
case INT32: |
|
return v.AsInt32() |
|
case INT64: |
|
return v.AsInt64() |
|
case UINT32: |
|
return v.AsUint32() |
|
case UINT64: |
|
return v.AsUint64() |
|
case FLOAT32: |
|
return v.AsFloat32() |
|
case FLOAT64: |
|
return v.AsFloat64() |
|
case STRING: |
|
return v.stringly |
|
} |
|
return unknownValueType{} |
|
} |
|
|
|
// Emit returns a string representation of Value's data. |
|
func (v Value) Emit() string { |
|
switch v.Type() { |
|
case ARRAY: |
|
return fmt.Sprint(v.array) |
|
case BOOL: |
|
return strconv.FormatBool(v.AsBool()) |
|
case INT32: |
|
return strconv.FormatInt(int64(v.AsInt32()), 10) |
|
case INT64: |
|
return strconv.FormatInt(v.AsInt64(), 10) |
|
case UINT32: |
|
return strconv.FormatUint(uint64(v.AsUint32()), 10) |
|
case UINT64: |
|
return strconv.FormatUint(v.AsUint64(), 10) |
|
case FLOAT32: |
|
return fmt.Sprint(v.AsFloat32()) |
|
case FLOAT64: |
|
return fmt.Sprint(v.AsFloat64()) |
|
case STRING: |
|
return v.stringly |
|
default: |
|
return "unknown" |
|
} |
|
} |
|
|
|
// MarshalJSON returns the JSON encoding of the Value. |
|
func (v Value) MarshalJSON() ([]byte, error) { |
|
var jsonVal struct { |
|
Type string |
|
Value interface{} |
|
} |
|
jsonVal.Type = v.Type().String() |
|
jsonVal.Value = v.AsInterface() |
|
return json.Marshal(jsonVal) |
|
}
|
|
|