forked from kubernetes/klog
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request kubernetes#287 from pohly/logger-call-depth-fix
fixes for PR kubernetes#280, refactoring, textlogger, unit test
- Loading branch information
Showing
18 changed files
with
2,427 additions
and
697 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,159 @@ | ||
// Copyright 2013 Google Inc. All Rights Reserved. | ||
// Copyright 2022 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 buffer provides a cache for byte.Buffer instances that can be reused | ||
// to avoid frequent allocation and deallocation. It also has utility code | ||
// for log header formatting that use these buffers. | ||
package buffer | ||
|
||
import ( | ||
"bytes" | ||
"os" | ||
"sync" | ||
"time" | ||
|
||
"k8s.io/klog/v2/internal/severity" | ||
) | ||
|
||
var ( | ||
// Pid is inserted into log headers. Can be overridden for tests. | ||
Pid = os.Getpid() | ||
) | ||
|
||
// Buffer holds a single byte.Buffer for reuse. The zero value is ready for | ||
// use. It also provides some helper methods for output formatting. | ||
type Buffer struct { | ||
bytes.Buffer | ||
Tmp [64]byte // temporary byte array for creating headers. | ||
next *Buffer | ||
} | ||
|
||
// Buffers manages the reuse of individual buffer instances. It is thread-safe. | ||
type Buffers struct { | ||
// mu protects the free list. It is separate from the main mutex | ||
// so buffers can be grabbed and printed to without holding the main lock, | ||
// for better parallelization. | ||
mu sync.Mutex | ||
|
||
// freeList is a list of byte buffers, maintained under mu. | ||
freeList *Buffer | ||
} | ||
|
||
// GetBuffer returns a new, ready-to-use buffer. | ||
func (bl *Buffers) GetBuffer() *Buffer { | ||
bl.mu.Lock() | ||
b := bl.freeList | ||
if b != nil { | ||
bl.freeList = b.next | ||
} | ||
bl.mu.Unlock() | ||
if b == nil { | ||
b = new(Buffer) | ||
} else { | ||
b.next = nil | ||
b.Reset() | ||
} | ||
return b | ||
} | ||
|
||
// PutBuffer returns a buffer to the free list. | ||
func (bl *Buffers) PutBuffer(b *Buffer) { | ||
if b.Len() >= 256 { | ||
// Let big buffers die a natural death. | ||
return | ||
} | ||
bl.mu.Lock() | ||
b.next = bl.freeList | ||
bl.freeList = b | ||
bl.mu.Unlock() | ||
} | ||
|
||
// Some custom tiny helper functions to print the log header efficiently. | ||
|
||
const digits = "0123456789" | ||
|
||
// twoDigits formats a zero-prefixed two-digit integer at buf.Tmp[i]. | ||
func (buf *Buffer) twoDigits(i, d int) { | ||
buf.Tmp[i+1] = digits[d%10] | ||
d /= 10 | ||
buf.Tmp[i] = digits[d%10] | ||
} | ||
|
||
// nDigits formats an n-digit integer at buf.Tmp[i], | ||
// padding with pad on the left. | ||
// It assumes d >= 0. | ||
func (buf *Buffer) nDigits(n, i, d int, pad byte) { | ||
j := n - 1 | ||
for ; j >= 0 && d > 0; j-- { | ||
buf.Tmp[i+j] = digits[d%10] | ||
d /= 10 | ||
} | ||
for ; j >= 0; j-- { | ||
buf.Tmp[i+j] = pad | ||
} | ||
} | ||
|
||
// someDigits formats a zero-prefixed variable-width integer at buf.Tmp[i]. | ||
func (buf *Buffer) someDigits(i, d int) int { | ||
// Print into the top, then copy down. We know there's space for at least | ||
// a 10-digit number. | ||
j := len(buf.Tmp) | ||
for { | ||
j-- | ||
buf.Tmp[j] = digits[d%10] | ||
d /= 10 | ||
if d == 0 { | ||
break | ||
} | ||
} | ||
return copy(buf.Tmp[i:], buf.Tmp[j:]) | ||
} | ||
|
||
// FormatHeader formats a log header using the provided file name and line number. | ||
func (buf *Buffer) FormatHeader(s severity.Severity, file string, line int, now time.Time) { | ||
if line < 0 { | ||
line = 0 // not a real line number, but acceptable to someDigits | ||
} | ||
if s > severity.FatalLog { | ||
s = severity.InfoLog // for safety. | ||
} | ||
|
||
// Avoid Fprintf, for speed. The format is so simple that we can do it quickly by hand. | ||
// It's worth about 3X. Fprintf is hard. | ||
_, month, day := now.Date() | ||
hour, minute, second := now.Clock() | ||
// Lmmdd hh:mm:ss.uuuuuu threadid file:line] | ||
buf.Tmp[0] = severity.Char[s] | ||
buf.twoDigits(1, int(month)) | ||
buf.twoDigits(3, day) | ||
buf.Tmp[5] = ' ' | ||
buf.twoDigits(6, hour) | ||
buf.Tmp[8] = ':' | ||
buf.twoDigits(9, minute) | ||
buf.Tmp[11] = ':' | ||
buf.twoDigits(12, second) | ||
buf.Tmp[14] = '.' | ||
buf.nDigits(6, 15, now.Nanosecond()/1000, '0') | ||
buf.Tmp[21] = ' ' | ||
buf.nDigits(7, 22, Pid, ' ') // TODO: should be TID | ||
buf.Tmp[29] = ' ' | ||
buf.Write(buf.Tmp[:30]) | ||
buf.WriteString(file) | ||
buf.Tmp[0] = ':' | ||
n := buf.someDigits(1, line) | ||
buf.Tmp[n+1] = ']' | ||
buf.Tmp[n+2] = ' ' | ||
buf.Write(buf.Tmp[:n+3]) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,211 @@ | ||
/* | ||
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 serialize | ||
|
||
import ( | ||
"bytes" | ||
"fmt" | ||
"strconv" | ||
) | ||
|
||
// WithValues implements LogSink.WithValues. The old key/value pairs are | ||
// assumed to be well-formed, the new ones are checked and padded if | ||
// necessary. It returns a new slice. | ||
func WithValues(oldKV, newKV []interface{}) []interface{} { | ||
if len(newKV) == 0 { | ||
return oldKV | ||
} | ||
newLen := len(oldKV) + len(newKV) | ||
hasMissingValue := newLen%2 != 0 | ||
if hasMissingValue { | ||
newLen++ | ||
} | ||
// The new LogSink must have its own slice. | ||
kv := make([]interface{}, 0, newLen) | ||
kv = append(kv, oldKV...) | ||
kv = append(kv, newKV...) | ||
if hasMissingValue { | ||
kv = append(kv, missingValue) | ||
} | ||
return kv | ||
} | ||
|
||
// TrimDuplicates deduplicates elements provided in multiple key/value tuple | ||
// slices, whilst maintaining the distinction between where the items are | ||
// contained. | ||
func TrimDuplicates(kvLists ...[]interface{}) [][]interface{} { | ||
// maintain a map of all seen keys | ||
seenKeys := map[interface{}]struct{}{} | ||
// build the same number of output slices as inputs | ||
outs := make([][]interface{}, len(kvLists)) | ||
// iterate over the input slices backwards, as 'later' kv specifications | ||
// of the same key will take precedence over earlier ones | ||
for i := len(kvLists) - 1; i >= 0; i-- { | ||
// initialise this output slice | ||
outs[i] = []interface{}{} | ||
// obtain a reference to the kvList we are processing | ||
// and make sure it has an even number of entries | ||
kvList := kvLists[i] | ||
if len(kvList)%2 != 0 { | ||
kvList = append(kvList, missingValue) | ||
} | ||
|
||
// start iterating at len(kvList) - 2 (i.e. the 2nd last item) for | ||
// slices that have an even number of elements. | ||
// We add (len(kvList) % 2) here to handle the case where there is an | ||
// odd number of elements in a kvList. | ||
// If there is an odd number, then the last element in the slice will | ||
// have the value 'null'. | ||
for i2 := len(kvList) - 2 + (len(kvList) % 2); i2 >= 0; i2 -= 2 { | ||
k := kvList[i2] | ||
// if we have already seen this key, do not include it again | ||
if _, ok := seenKeys[k]; ok { | ||
continue | ||
} | ||
// make a note that we've observed a new key | ||
seenKeys[k] = struct{}{} | ||
// attempt to obtain the value of the key | ||
var v interface{} | ||
// i2+1 should only ever be out of bounds if we handling the first | ||
// iteration over a slice with an odd number of elements | ||
if i2+1 < len(kvList) { | ||
v = kvList[i2+1] | ||
} | ||
// add this KV tuple to the *start* of the output list to maintain | ||
// the original order as we are iterating over the slice backwards | ||
outs[i] = append([]interface{}{k, v}, outs[i]...) | ||
} | ||
} | ||
return outs | ||
} | ||
|
||
const missingValue = "(MISSING)" | ||
|
||
// KVListFormat serializes all key/value pairs into the provided buffer. | ||
// A space gets inserted before the first pair and between each pair. | ||
func KVListFormat(b *bytes.Buffer, keysAndValues ...interface{}) { | ||
for i := 0; i < len(keysAndValues); i += 2 { | ||
var v interface{} | ||
k := keysAndValues[i] | ||
if i+1 < len(keysAndValues) { | ||
v = keysAndValues[i+1] | ||
} else { | ||
v = missingValue | ||
} | ||
b.WriteByte(' ') | ||
// Keys are assumed to be well-formed according to | ||
// https://github.com/kubernetes/community/blob/master/contributors/devel/sig-instrumentation/migration-to-structured-logging.md#name-arguments | ||
// for the sake of performance. Keys with spaces, | ||
// special characters, etc. will break parsing. | ||
if k, ok := k.(string); ok { | ||
// Avoid one allocation when the key is a string, which | ||
// normally it should be. | ||
b.WriteString(k) | ||
} else { | ||
b.WriteString(fmt.Sprintf("%s", k)) | ||
} | ||
|
||
// The type checks are sorted so that more frequently used ones | ||
// come first because that is then faster in the common | ||
// cases. In Kubernetes, ObjectRef (a Stringer) is more common | ||
// than plain strings | ||
// (https://github.com/kubernetes/kubernetes/pull/106594#issuecomment-975526235). | ||
switch v := v.(type) { | ||
case fmt.Stringer: | ||
writeStringValue(b, true, stringerToString(v)) | ||
case string: | ||
writeStringValue(b, true, v) | ||
case error: | ||
writeStringValue(b, true, v.Error()) | ||
case []byte: | ||
// In https://github.com/kubernetes/klog/pull/237 it was decided | ||
// to format byte slices with "%+q". The advantages of that are: | ||
// - readable output if the bytes happen to be printable | ||
// - non-printable bytes get represented as unicode escape | ||
// sequences (\uxxxx) | ||
// | ||
// The downsides are that we cannot use the faster | ||
// strconv.Quote here and that multi-line output is not | ||
// supported. If developers know that a byte array is | ||
// printable and they want multi-line output, they can | ||
// convert the value to string before logging it. | ||
b.WriteByte('=') | ||
b.WriteString(fmt.Sprintf("%+q", v)) | ||
default: | ||
writeStringValue(b, false, fmt.Sprintf("%+v", v)) | ||
} | ||
} | ||
} | ||
|
||
func stringerToString(s fmt.Stringer) (ret string) { | ||
defer func() { | ||
if err := recover(); err != nil { | ||
ret = "nil" | ||
} | ||
}() | ||
ret = s.String() | ||
return | ||
} | ||
|
||
func writeStringValue(b *bytes.Buffer, quote bool, v string) { | ||
data := []byte(v) | ||
index := bytes.IndexByte(data, '\n') | ||
if index == -1 { | ||
b.WriteByte('=') | ||
if quote { | ||
// Simple string, quote quotation marks and non-printable characters. | ||
b.WriteString(strconv.Quote(v)) | ||
return | ||
} | ||
// Non-string with no line breaks. | ||
b.WriteString(v) | ||
return | ||
} | ||
|
||
// Complex multi-line string, show as-is with indention like this: | ||
// I... "hello world" key=< | ||
// <tab>line 1 | ||
// <tab>line 2 | ||
// > | ||
// | ||
// Tabs indent the lines of the value while the end of string delimiter | ||
// is indented with a space. That has two purposes: | ||
// - visual difference between the two for a human reader because indention | ||
// will be different | ||
// - no ambiguity when some value line starts with the end delimiter | ||
// | ||
// One downside is that the output cannot distinguish between strings that | ||
// end with a line break and those that don't because the end delimiter | ||
// will always be on the next line. | ||
b.WriteString("=<\n") | ||
for index != -1 { | ||
b.WriteByte('\t') | ||
b.Write(data[0 : index+1]) | ||
data = data[index+1:] | ||
index = bytes.IndexByte(data, '\n') | ||
} | ||
if len(data) == 0 { | ||
// String ended with line break, don't add another. | ||
b.WriteString(" >") | ||
} else { | ||
// No line break at end of last line, write rest of string and | ||
// add one. | ||
b.WriteByte('\t') | ||
b.Write(data) | ||
b.WriteString("\n >") | ||
} | ||
} |
Oops, something went wrong.