mirror of https://github.com/tikv/client-go.git
178 lines
5.2 KiB
Go
178 lines
5.2 KiB
Go
// Copyright 2021 TiKV 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.
|
|
|
|
// NOTE: The code in this file is based on code from the
|
|
// TiDB project, licensed under the Apache License v 2.0
|
|
//
|
|
// https://github.com/pingcap/tidb/tree/cc5e161ac06827589c4966674597c137cc9e809c/store/tikv/util/misc.go
|
|
//
|
|
|
|
// Copyright 2021 PingCAP, Inc.
|
|
//
|
|
// 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 util
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/tikv/client-go/v2/internal/logutil"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
// GCTimeFormat is the format that gc_worker used to store times.
|
|
const GCTimeFormat = "20060102-15:04:05.000 -0700"
|
|
|
|
// gcTimeFormatOld is the format that gc_worker used to store times before, for compatibility we keep it.
|
|
const gcTimeFormatOld = "20060102-15:04:05 -0700"
|
|
|
|
// CompatibleParseGCTime parses a string with `GCTimeFormat` and returns a time.Time. If `value` can't be parsed as that
|
|
// format, truncate to last space and try again. This function is only useful when loading times that saved by
|
|
// gc_worker. We have changed the format that gc_worker saves time (removed the last field), but when loading times it
|
|
// should be compatible with the old format.
|
|
func CompatibleParseGCTime(value string) (time.Time, error) {
|
|
// The old format could parse the value with new format,
|
|
// let `GCTimeFormat` only be used when store the time.
|
|
t, err := time.Parse(gcTimeFormatOld, value)
|
|
if err != nil {
|
|
// Remove the last field that separated by space
|
|
parts := strings.Split(value, " ")
|
|
prefix := strings.Join(parts[:len(parts)-1], " ")
|
|
t, err = time.Parse(gcTimeFormatOld, prefix)
|
|
}
|
|
|
|
if err != nil {
|
|
err = errors.Errorf("string \"%v\" doesn't has a prefix that matches format \"%v\"", value, GCTimeFormat)
|
|
}
|
|
return t, err
|
|
}
|
|
|
|
// WithRecovery wraps goroutine startup call with force recovery.
|
|
// it will dump current goroutine stack into log if catch any recover result.
|
|
//
|
|
// exec: execute logic function.
|
|
// recoverFn: handler will be called after recover and before dump stack, passing `nil` means noop.
|
|
func WithRecovery(exec func(), recoverFn func(r interface{})) {
|
|
defer func() {
|
|
r := recover()
|
|
if recoverFn != nil {
|
|
recoverFn(r)
|
|
}
|
|
if r != nil {
|
|
logutil.BgLogger().Error("panic in the recoverable goroutine",
|
|
zap.Any("r", r),
|
|
zap.Stack("stack trace"))
|
|
}
|
|
}()
|
|
exec()
|
|
}
|
|
|
|
type sessionIDCtxKey struct{}
|
|
|
|
// SessionID is the context key type to mark a session.
|
|
var SessionID = sessionIDCtxKey{}
|
|
|
|
// SetSessionID sets session id into context
|
|
func SetSessionID(ctx context.Context, sessionID uint64) context.Context {
|
|
return context.WithValue(ctx, SessionID, sessionID)
|
|
}
|
|
|
|
const (
|
|
byteSizeGB = int64(1 << 30)
|
|
byteSizeMB = int64(1 << 20)
|
|
byteSizeKB = int64(1 << 10)
|
|
byteSizeBB = int64(1)
|
|
)
|
|
|
|
// FormatBytes uses to format bytes, this function will prune precision before format bytes.
|
|
func FormatBytes(numBytes int64) string {
|
|
if numBytes <= byteSizeKB {
|
|
return BytesToString(numBytes)
|
|
}
|
|
unit, unitStr := getByteUnit(numBytes)
|
|
if unit == byteSizeBB {
|
|
return BytesToString(numBytes)
|
|
}
|
|
v := float64(numBytes) / float64(unit)
|
|
decimal := 1
|
|
if numBytes%unit == 0 {
|
|
decimal = 0
|
|
} else if v < 10 {
|
|
decimal = 2
|
|
}
|
|
return fmt.Sprintf("%v %s", strconv.FormatFloat(v, 'f', decimal, 64), unitStr)
|
|
}
|
|
|
|
func getByteUnit(b int64) (int64, string) {
|
|
if b > byteSizeGB {
|
|
return byteSizeGB, "GB"
|
|
} else if b > byteSizeMB {
|
|
return byteSizeMB, "MB"
|
|
} else if b > byteSizeKB {
|
|
return byteSizeKB, "KB"
|
|
}
|
|
return byteSizeBB, "Bytes"
|
|
}
|
|
|
|
// BytesToString converts the memory consumption to a readable string.
|
|
func BytesToString(numBytes int64) string {
|
|
GB := float64(numBytes) / float64(byteSizeGB)
|
|
if GB > 1 {
|
|
return fmt.Sprintf("%v GB", GB)
|
|
}
|
|
|
|
MB := float64(numBytes) / float64(byteSizeMB)
|
|
if MB > 1 {
|
|
return fmt.Sprintf("%v MB", MB)
|
|
}
|
|
|
|
KB := float64(numBytes) / float64(byteSizeKB)
|
|
if KB > 1 {
|
|
return fmt.Sprintf("%v KB", KB)
|
|
}
|
|
|
|
return fmt.Sprintf("%v Bytes", numBytes)
|
|
}
|
|
|
|
type Option[T interface{}] struct {
|
|
inner *T
|
|
}
|
|
|
|
func Some[T interface{}](inner T) Option[T] {
|
|
return Option[T]{inner: &inner}
|
|
}
|
|
|
|
func None[T interface{}]() Option[T] {
|
|
return Option[T]{inner: nil}
|
|
}
|
|
|
|
func (o Option[T]) Inner() *T {
|
|
return o.inner
|
|
}
|