mirror of
https://gitee.com/johng/gf
synced 2026-06-07 02:12:11 +08:00
This PR includes the following changes: - **Upgrade `.golangci.yml`**: Updated the configuration file to align with the latest golangci-lint version, ensuring compatibility and leveraging new features. - **Refactor GitHub Action workflow**: Modified `golangci-lint.yml` in the GitHub Actions workflow to reflect the updated configuration and improve CI performance. - **Codebase optimization**: Refactored code to address issues and warnings raised by the updated golangci-lint rules, including: - Improved function length and complexity. - Enhanced error handling and variable naming conventions. - Fixed minor issues such as unused imports and formatting inconsistencies. These changes aim to maintain code quality, ensure compatibility with the latest tools, and improve overall maintainability.
502 lines
12 KiB
Go
502 lines
12 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package gutil
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/gogf/gf/v2/internal/reflection"
|
|
"github.com/gogf/gf/v2/os/gstructs"
|
|
"github.com/gogf/gf/v2/text/gstr"
|
|
)
|
|
|
|
// iString is used for type assert api for String().
|
|
type iString interface {
|
|
String() string
|
|
}
|
|
|
|
// iError is used for type assert api for Error().
|
|
type iError interface {
|
|
Error() string
|
|
}
|
|
|
|
// iMarshalJSON is the interface for custom Json marshaling.
|
|
type iMarshalJSON interface {
|
|
MarshalJSON() ([]byte, error)
|
|
}
|
|
|
|
// DumpOption specifies the behavior of function Export.
|
|
type DumpOption struct {
|
|
WithType bool // WithType specifies dumping content with type information.
|
|
ExportedOnly bool // Only dump Exported fields for structs.
|
|
}
|
|
|
|
// Dump prints variables `values` to stdout with more manually readable.
|
|
func Dump(values ...interface{}) {
|
|
for _, value := range values {
|
|
DumpWithOption(value, DumpOption{
|
|
WithType: false,
|
|
ExportedOnly: false,
|
|
})
|
|
}
|
|
}
|
|
|
|
// DumpWithType acts like Dump, but with type information.
|
|
// Also see Dump.
|
|
func DumpWithType(values ...interface{}) {
|
|
for _, value := range values {
|
|
DumpWithOption(value, DumpOption{
|
|
WithType: true,
|
|
ExportedOnly: false,
|
|
})
|
|
}
|
|
}
|
|
|
|
// DumpWithOption returns variables `values` as a string with more manually readable.
|
|
func DumpWithOption(value interface{}, option DumpOption) {
|
|
buffer := bytes.NewBuffer(nil)
|
|
DumpTo(buffer, value, DumpOption{
|
|
WithType: option.WithType,
|
|
ExportedOnly: option.ExportedOnly,
|
|
})
|
|
fmt.Println(buffer.String())
|
|
}
|
|
|
|
// DumpTo writes variables `values` as a string in to `writer` with more manually readable
|
|
func DumpTo(writer io.Writer, value interface{}, option DumpOption) {
|
|
buffer := bytes.NewBuffer(nil)
|
|
doDump(value, "", buffer, doDumpOption{
|
|
WithType: option.WithType,
|
|
ExportedOnly: option.ExportedOnly,
|
|
})
|
|
_, _ = writer.Write(buffer.Bytes())
|
|
}
|
|
|
|
type doDumpOption struct {
|
|
WithType bool
|
|
ExportedOnly bool
|
|
DumpedPointerSet map[string]struct{}
|
|
}
|
|
|
|
func doDump(value interface{}, indent string, buffer *bytes.Buffer, option doDumpOption) {
|
|
if option.DumpedPointerSet == nil {
|
|
option.DumpedPointerSet = map[string]struct{}{}
|
|
}
|
|
|
|
if value == nil {
|
|
buffer.WriteString(`<nil>`)
|
|
return
|
|
}
|
|
var reflectValue reflect.Value
|
|
if v, ok := value.(reflect.Value); ok {
|
|
reflectValue = v
|
|
if v.IsValid() && v.CanInterface() {
|
|
value = v.Interface()
|
|
} else {
|
|
if convertedValue, ok := reflection.ValueToInterface(v); ok {
|
|
value = convertedValue
|
|
}
|
|
}
|
|
} else {
|
|
reflectValue = reflect.ValueOf(value)
|
|
}
|
|
var reflectKind = reflectValue.Kind()
|
|
// Double check nil value.
|
|
if value == nil || reflectKind == reflect.Invalid {
|
|
buffer.WriteString(`<nil>`)
|
|
return
|
|
}
|
|
var (
|
|
reflectTypeName = reflectValue.Type().String()
|
|
ptrAddress string
|
|
newIndent = indent + dumpIndent
|
|
)
|
|
reflectTypeName = strings.ReplaceAll(reflectTypeName, `[]uint8`, `[]byte`)
|
|
for reflectKind == reflect.Ptr {
|
|
if ptrAddress == "" {
|
|
ptrAddress = fmt.Sprintf(`0x%x`, reflectValue.Pointer())
|
|
}
|
|
reflectValue = reflectValue.Elem()
|
|
reflectKind = reflectValue.Kind()
|
|
}
|
|
var (
|
|
exportInternalInput = doDumpInternalInput{
|
|
Value: value,
|
|
Indent: indent,
|
|
NewIndent: newIndent,
|
|
Buffer: buffer,
|
|
Option: option,
|
|
PtrAddress: ptrAddress,
|
|
ReflectValue: reflectValue,
|
|
ReflectTypeName: reflectTypeName,
|
|
ExportedOnly: option.ExportedOnly,
|
|
DumpedPointerSet: option.DumpedPointerSet,
|
|
}
|
|
)
|
|
switch reflectKind {
|
|
case reflect.Slice, reflect.Array:
|
|
doDumpSlice(exportInternalInput)
|
|
|
|
case reflect.Map:
|
|
doDumpMap(exportInternalInput)
|
|
|
|
case reflect.Struct:
|
|
doDumpStruct(exportInternalInput)
|
|
|
|
case reflect.String:
|
|
doDumpString(exportInternalInput)
|
|
|
|
case reflect.Bool:
|
|
doDumpBool(exportInternalInput)
|
|
|
|
case
|
|
reflect.Int,
|
|
reflect.Int8,
|
|
reflect.Int16,
|
|
reflect.Int32,
|
|
reflect.Int64,
|
|
reflect.Uint,
|
|
reflect.Uint8,
|
|
reflect.Uint16,
|
|
reflect.Uint32,
|
|
reflect.Uint64,
|
|
reflect.Float32,
|
|
reflect.Float64,
|
|
reflect.Complex64,
|
|
reflect.Complex128:
|
|
doDumpNumber(exportInternalInput)
|
|
|
|
case reflect.Chan:
|
|
buffer.WriteString(fmt.Sprintf(`<%s>`, reflectValue.Type().String()))
|
|
|
|
case reflect.Func:
|
|
if reflectValue.IsNil() || !reflectValue.IsValid() {
|
|
buffer.WriteString(`<nil>`)
|
|
} else {
|
|
buffer.WriteString(fmt.Sprintf(`<%s>`, reflectValue.Type().String()))
|
|
}
|
|
|
|
case reflect.Interface:
|
|
doDump(exportInternalInput.ReflectValue.Elem(), indent, buffer, option)
|
|
|
|
default:
|
|
doDumpDefault(exportInternalInput)
|
|
}
|
|
}
|
|
|
|
type doDumpInternalInput struct {
|
|
Value interface{}
|
|
Indent string
|
|
NewIndent string
|
|
Buffer *bytes.Buffer
|
|
Option doDumpOption
|
|
ReflectValue reflect.Value
|
|
ReflectTypeName string
|
|
PtrAddress string
|
|
ExportedOnly bool
|
|
DumpedPointerSet map[string]struct{}
|
|
}
|
|
|
|
func doDumpSlice(in doDumpInternalInput) {
|
|
if b, ok := in.Value.([]byte); ok {
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(fmt.Sprintf(`"%s"`, addSlashesForString(string(b))))
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
`%s(%d) "%s"`,
|
|
in.ReflectTypeName,
|
|
len(string(b)),
|
|
string(b),
|
|
))
|
|
}
|
|
return
|
|
}
|
|
if in.ReflectValue.Len() == 0 {
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString("[]")
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(0) []", in.ReflectTypeName))
|
|
}
|
|
return
|
|
}
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString("[\n")
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(%d) [\n", in.ReflectTypeName, in.ReflectValue.Len()))
|
|
}
|
|
for i := 0; i < in.ReflectValue.Len(); i++ {
|
|
in.Buffer.WriteString(in.NewIndent)
|
|
doDump(in.ReflectValue.Index(i), in.NewIndent, in.Buffer, in.Option)
|
|
in.Buffer.WriteString(",\n")
|
|
}
|
|
in.Buffer.WriteString(fmt.Sprintf("%s]", in.Indent))
|
|
}
|
|
|
|
func doDumpMap(in doDumpInternalInput) {
|
|
var mapKeys = make([]reflect.Value, 0)
|
|
for _, key := range in.ReflectValue.MapKeys() {
|
|
if !key.CanInterface() {
|
|
continue
|
|
}
|
|
mapKey := key
|
|
mapKeys = append(mapKeys, mapKey)
|
|
}
|
|
if len(mapKeys) == 0 {
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString("{}")
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(0) {}", in.ReflectTypeName))
|
|
}
|
|
return
|
|
}
|
|
var (
|
|
maxSpaceNum = 0
|
|
tmpSpaceNum = 0
|
|
mapKeyStr = ""
|
|
)
|
|
for _, key := range mapKeys {
|
|
tmpSpaceNum = len(fmt.Sprintf(`%v`, key.Interface()))
|
|
if tmpSpaceNum > maxSpaceNum {
|
|
maxSpaceNum = tmpSpaceNum
|
|
}
|
|
}
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString("{\n")
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(%d) {\n", in.ReflectTypeName, len(mapKeys)))
|
|
}
|
|
for _, mapKey := range mapKeys {
|
|
tmpSpaceNum = len(fmt.Sprintf(`%v`, mapKey.Interface()))
|
|
if mapKey.Kind() == reflect.String {
|
|
mapKeyStr = fmt.Sprintf(`"%v"`, mapKey.Interface())
|
|
} else {
|
|
mapKeyStr = fmt.Sprintf(`%v`, mapKey.Interface())
|
|
}
|
|
// Map key and indent string dump.
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
"%s%v:%s",
|
|
in.NewIndent,
|
|
mapKeyStr,
|
|
strings.Repeat(" ", maxSpaceNum-tmpSpaceNum+1),
|
|
))
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
"%s%s(%v):%s",
|
|
in.NewIndent,
|
|
mapKey.Type().String(),
|
|
mapKeyStr,
|
|
strings.Repeat(" ", maxSpaceNum-tmpSpaceNum+1),
|
|
))
|
|
}
|
|
// Map value dump.
|
|
doDump(in.ReflectValue.MapIndex(mapKey), in.NewIndent, in.Buffer, in.Option)
|
|
in.Buffer.WriteString(",\n")
|
|
}
|
|
in.Buffer.WriteString(fmt.Sprintf("%s}", in.Indent))
|
|
}
|
|
|
|
func doDumpStruct(in doDumpInternalInput) {
|
|
if in.PtrAddress != "" {
|
|
if _, ok := in.DumpedPointerSet[in.PtrAddress]; ok {
|
|
in.Buffer.WriteString(fmt.Sprintf(`<cycle dump %s>`, in.PtrAddress))
|
|
return
|
|
}
|
|
}
|
|
in.DumpedPointerSet[in.PtrAddress] = struct{}{}
|
|
|
|
structFields, _ := gstructs.Fields(gstructs.FieldsInput{
|
|
Pointer: in.Value,
|
|
RecursiveOption: gstructs.RecursiveOptionEmbedded,
|
|
})
|
|
var (
|
|
hasNoExportedFields = true
|
|
_, isReflectValue = in.Value.(reflect.Value)
|
|
)
|
|
for _, field := range structFields {
|
|
if field.IsExported() {
|
|
hasNoExportedFields = false
|
|
break
|
|
}
|
|
}
|
|
if !isReflectValue && (len(structFields) == 0 || hasNoExportedFields) {
|
|
var (
|
|
structContentStr = ""
|
|
attributeCountStr = "0"
|
|
)
|
|
if v, ok := in.Value.(iString); ok {
|
|
structContentStr = v.String()
|
|
} else if v, ok := in.Value.(iError); ok {
|
|
structContentStr = v.Error()
|
|
} else if v, ok := in.Value.(iMarshalJSON); ok {
|
|
b, _ := v.MarshalJSON()
|
|
structContentStr = string(b)
|
|
} else {
|
|
// Has no common interface implements.
|
|
if len(structFields) != 0 {
|
|
goto dumpStructFields
|
|
}
|
|
}
|
|
if structContentStr == "" {
|
|
structContentStr = "{}"
|
|
} else {
|
|
structContentStr = fmt.Sprintf(`"%s"`, addSlashesForString(structContentStr))
|
|
attributeCountStr = fmt.Sprintf(`%d`, len(structContentStr)-2)
|
|
}
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(structContentStr)
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
"%s(%s) %s",
|
|
in.ReflectTypeName,
|
|
attributeCountStr,
|
|
structContentStr,
|
|
))
|
|
}
|
|
return
|
|
}
|
|
|
|
dumpStructFields:
|
|
var (
|
|
maxSpaceNum = 0
|
|
tmpSpaceNum = 0
|
|
)
|
|
for _, field := range structFields {
|
|
if in.ExportedOnly && !field.IsExported() {
|
|
continue
|
|
}
|
|
tmpSpaceNum = len(field.Name())
|
|
if tmpSpaceNum > maxSpaceNum {
|
|
maxSpaceNum = tmpSpaceNum
|
|
}
|
|
}
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString("{\n")
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(%d) {\n", in.ReflectTypeName, len(structFields)))
|
|
}
|
|
for _, field := range structFields {
|
|
if in.ExportedOnly && !field.IsExported() {
|
|
continue
|
|
}
|
|
tmpSpaceNum = len(fmt.Sprintf(`%v`, field.Name()))
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
"%s%s:%s",
|
|
in.NewIndent,
|
|
field.Name(),
|
|
strings.Repeat(" ", maxSpaceNum-tmpSpaceNum+1),
|
|
))
|
|
doDump(field.Value, in.NewIndent, in.Buffer, in.Option)
|
|
in.Buffer.WriteString(",\n")
|
|
}
|
|
in.Buffer.WriteString(fmt.Sprintf("%s}", in.Indent))
|
|
}
|
|
|
|
func doDumpNumber(in doDumpInternalInput) {
|
|
if v, ok := in.Value.(iString); ok {
|
|
s := v.String()
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(fmt.Sprintf(`"%v"`, addSlashesForString(s)))
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
`%s(%d) "%v"`,
|
|
in.ReflectTypeName,
|
|
len(s),
|
|
addSlashesForString(s),
|
|
))
|
|
}
|
|
} else {
|
|
doDumpDefault(in)
|
|
}
|
|
}
|
|
|
|
func doDumpString(in doDumpInternalInput) {
|
|
s := in.ReflectValue.String()
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(fmt.Sprintf(`"%v"`, addSlashesForString(s)))
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf(
|
|
`%s(%d) "%v"`,
|
|
in.ReflectTypeName,
|
|
len(s),
|
|
addSlashesForString(s),
|
|
))
|
|
}
|
|
}
|
|
|
|
func doDumpBool(in doDumpInternalInput) {
|
|
var s string
|
|
if in.ReflectValue.Bool() {
|
|
s = `true`
|
|
} else {
|
|
s = `false`
|
|
}
|
|
if in.Option.WithType {
|
|
s = fmt.Sprintf(`bool(%s)`, s)
|
|
}
|
|
in.Buffer.WriteString(s)
|
|
}
|
|
|
|
func doDumpDefault(in doDumpInternalInput) {
|
|
var s string
|
|
if in.ReflectValue.IsValid() && in.ReflectValue.CanInterface() {
|
|
s = fmt.Sprintf("%v", in.ReflectValue.Interface())
|
|
}
|
|
if s == "" {
|
|
s = fmt.Sprintf("%v", in.Value)
|
|
}
|
|
s = gstr.Trim(s, `<>`)
|
|
if !in.Option.WithType {
|
|
in.Buffer.WriteString(s)
|
|
} else {
|
|
in.Buffer.WriteString(fmt.Sprintf("%s(%s)", in.ReflectTypeName, s))
|
|
}
|
|
}
|
|
|
|
func addSlashesForString(s string) string {
|
|
return gstr.ReplaceByMap(s, map[string]string{
|
|
`"`: `\"`,
|
|
`'`: `\'`,
|
|
"\r": `\r`,
|
|
"\t": `\t`,
|
|
"\n": `\n`,
|
|
})
|
|
}
|
|
|
|
// DumpJson pretty dumps json content to stdout.
|
|
func DumpJson(value any) {
|
|
switch result := value.(type) {
|
|
case []byte:
|
|
doDumpJSON(result)
|
|
case string:
|
|
doDumpJSON([]byte(result))
|
|
default:
|
|
jsonContent, err := json.Marshal(value)
|
|
if err != nil {
|
|
fmt.Println(err.Error())
|
|
return
|
|
}
|
|
doDumpJSON(jsonContent)
|
|
}
|
|
}
|
|
|
|
func doDumpJSON(jsonContent []byte) {
|
|
var (
|
|
buffer = bytes.NewBuffer(nil)
|
|
jsonBytes = jsonContent
|
|
)
|
|
if err := json.Indent(buffer, jsonBytes, "", " "); err != nil {
|
|
fmt.Println(err.Error())
|
|
}
|
|
fmt.Println(buffer.String())
|
|
}
|