mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-22 00:00:39 +01:00
213 lines
5.3 KiB
Go
213 lines
5.3 KiB
Go
|
/*
|
||
|
Copyright 2021 The Kubernetes 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 klog
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"fmt"
|
||
|
"reflect"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/go-logr/logr"
|
||
|
)
|
||
|
|
||
|
// ObjectRef references a kubernetes object
|
||
|
type ObjectRef struct {
|
||
|
Name string `json:"name"`
|
||
|
Namespace string `json:"namespace,omitempty"`
|
||
|
}
|
||
|
|
||
|
func (ref ObjectRef) String() string {
|
||
|
if ref.Namespace != "" {
|
||
|
var builder strings.Builder
|
||
|
builder.Grow(len(ref.Namespace) + len(ref.Name) + 1)
|
||
|
builder.WriteString(ref.Namespace)
|
||
|
builder.WriteRune('/')
|
||
|
builder.WriteString(ref.Name)
|
||
|
return builder.String()
|
||
|
}
|
||
|
return ref.Name
|
||
|
}
|
||
|
|
||
|
func (ref ObjectRef) WriteText(out *bytes.Buffer) {
|
||
|
out.WriteRune('"')
|
||
|
ref.writeUnquoted(out)
|
||
|
out.WriteRune('"')
|
||
|
}
|
||
|
|
||
|
func (ref ObjectRef) writeUnquoted(out *bytes.Buffer) {
|
||
|
if ref.Namespace != "" {
|
||
|
out.WriteString(ref.Namespace)
|
||
|
out.WriteRune('/')
|
||
|
}
|
||
|
out.WriteString(ref.Name)
|
||
|
}
|
||
|
|
||
|
// MarshalLog ensures that loggers with support for structured output will log
|
||
|
// as a struct by removing the String method via a custom type.
|
||
|
func (ref ObjectRef) MarshalLog() interface{} {
|
||
|
type or ObjectRef
|
||
|
return or(ref)
|
||
|
}
|
||
|
|
||
|
var _ logr.Marshaler = ObjectRef{}
|
||
|
|
||
|
// KMetadata is a subset of the kubernetes k8s.io/apimachinery/pkg/apis/meta/v1.Object interface
|
||
|
// this interface may expand in the future, but will always be a subset of the
|
||
|
// kubernetes k8s.io/apimachinery/pkg/apis/meta/v1.Object interface
|
||
|
type KMetadata interface {
|
||
|
GetName() string
|
||
|
GetNamespace() string
|
||
|
}
|
||
|
|
||
|
// KObj returns ObjectRef from ObjectMeta
|
||
|
func KObj(obj KMetadata) ObjectRef {
|
||
|
if obj == nil {
|
||
|
return ObjectRef{}
|
||
|
}
|
||
|
if val := reflect.ValueOf(obj); val.Kind() == reflect.Ptr && val.IsNil() {
|
||
|
return ObjectRef{}
|
||
|
}
|
||
|
|
||
|
return ObjectRef{
|
||
|
Name: obj.GetName(),
|
||
|
Namespace: obj.GetNamespace(),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// KRef returns ObjectRef from name and namespace
|
||
|
func KRef(namespace, name string) ObjectRef {
|
||
|
return ObjectRef{
|
||
|
Name: name,
|
||
|
Namespace: namespace,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// KObjs returns slice of ObjectRef from an slice of ObjectMeta
|
||
|
//
|
||
|
// DEPRECATED: Use KObjSlice instead, it has better performance.
|
||
|
func KObjs(arg interface{}) []ObjectRef {
|
||
|
s := reflect.ValueOf(arg)
|
||
|
if s.Kind() != reflect.Slice {
|
||
|
return nil
|
||
|
}
|
||
|
objectRefs := make([]ObjectRef, 0, s.Len())
|
||
|
for i := 0; i < s.Len(); i++ {
|
||
|
if v, ok := s.Index(i).Interface().(KMetadata); ok {
|
||
|
objectRefs = append(objectRefs, KObj(v))
|
||
|
} else {
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
return objectRefs
|
||
|
}
|
||
|
|
||
|
// KObjSlice takes a slice of objects that implement the KMetadata interface
|
||
|
// and returns an object that gets logged as a slice of ObjectRef values or a
|
||
|
// string containing those values, depending on whether the logger prefers text
|
||
|
// output or structured output.
|
||
|
//
|
||
|
// An error string is logged when KObjSlice is not passed a suitable slice.
|
||
|
//
|
||
|
// Processing of the argument is delayed until the value actually gets logged,
|
||
|
// in contrast to KObjs where that overhead is incurred regardless of whether
|
||
|
// the result is needed.
|
||
|
func KObjSlice(arg interface{}) interface{} {
|
||
|
return kobjSlice{arg: arg}
|
||
|
}
|
||
|
|
||
|
type kobjSlice struct {
|
||
|
arg interface{}
|
||
|
}
|
||
|
|
||
|
var _ fmt.Stringer = kobjSlice{}
|
||
|
var _ logr.Marshaler = kobjSlice{}
|
||
|
|
||
|
func (ks kobjSlice) String() string {
|
||
|
objectRefs, errStr := ks.process()
|
||
|
if errStr != "" {
|
||
|
return errStr
|
||
|
}
|
||
|
return fmt.Sprintf("%v", objectRefs)
|
||
|
}
|
||
|
|
||
|
func (ks kobjSlice) MarshalLog() interface{} {
|
||
|
objectRefs, errStr := ks.process()
|
||
|
if errStr != "" {
|
||
|
return errStr
|
||
|
}
|
||
|
return objectRefs
|
||
|
}
|
||
|
|
||
|
func (ks kobjSlice) process() (objs []interface{}, err string) {
|
||
|
s := reflect.ValueOf(ks.arg)
|
||
|
switch s.Kind() {
|
||
|
case reflect.Invalid:
|
||
|
// nil parameter, print as nil.
|
||
|
return nil, ""
|
||
|
case reflect.Slice:
|
||
|
// Okay, handle below.
|
||
|
default:
|
||
|
return nil, fmt.Sprintf("<KObjSlice needs a slice, got type %T>", ks.arg)
|
||
|
}
|
||
|
objectRefs := make([]interface{}, 0, s.Len())
|
||
|
for i := 0; i < s.Len(); i++ {
|
||
|
item := s.Index(i).Interface()
|
||
|
if item == nil {
|
||
|
objectRefs = append(objectRefs, nil)
|
||
|
} else if v, ok := item.(KMetadata); ok {
|
||
|
objectRefs = append(objectRefs, KObj(v))
|
||
|
} else {
|
||
|
return nil, fmt.Sprintf("<KObjSlice needs a slice of values implementing KMetadata, got type %T>", item)
|
||
|
}
|
||
|
}
|
||
|
return objectRefs, ""
|
||
|
}
|
||
|
|
||
|
var nilToken = []byte("null")
|
||
|
|
||
|
func (ks kobjSlice) WriteText(out *bytes.Buffer) {
|
||
|
s := reflect.ValueOf(ks.arg)
|
||
|
switch s.Kind() {
|
||
|
case reflect.Invalid:
|
||
|
// nil parameter, print as null.
|
||
|
out.Write(nilToken)
|
||
|
return
|
||
|
case reflect.Slice:
|
||
|
// Okay, handle below.
|
||
|
default:
|
||
|
fmt.Fprintf(out, `"<KObjSlice needs a slice, got type %T>"`, ks.arg)
|
||
|
return
|
||
|
}
|
||
|
out.Write([]byte{'['})
|
||
|
defer out.Write([]byte{']'})
|
||
|
for i := 0; i < s.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
out.Write([]byte{','})
|
||
|
}
|
||
|
item := s.Index(i).Interface()
|
||
|
if item == nil {
|
||
|
out.Write(nilToken)
|
||
|
} else if v, ok := item.(KMetadata); ok {
|
||
|
KObj(v).WriteText(out)
|
||
|
} else {
|
||
|
fmt.Fprintf(out, `"<KObjSlice needs a slice of values implementing KMetadata, got type %T>"`, item)
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|