2023-07-21 17:07:04 +08:00
|
|
|
|
package structx
|
2020-09-01 10:34:11 +08:00
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"errors"
|
|
|
|
|
|
"fmt"
|
2023-10-12 12:14:56 +08:00
|
|
|
|
"mayfly-go/pkg/utils/collx"
|
2020-09-01 10:34:11 +08:00
|
|
|
|
"reflect"
|
|
|
|
|
|
"strconv"
|
|
|
|
|
|
"strings"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-05-24 16:22:54 +08:00
|
|
|
|
// CopyTo 将fromValue转为T类型并返回
|
|
|
|
|
|
func CopyTo[T any](fromValue any) T {
|
|
|
|
|
|
t := NewInstance[T]()
|
|
|
|
|
|
Copy(t, fromValue)
|
|
|
|
|
|
return t
|
|
|
|
|
|
}
|
2020-09-01 10:34:11 +08:00
|
|
|
|
|
2025-05-24 16:22:54 +08:00
|
|
|
|
// CopySliceTo 将fromValue转为[]T类型并返回
|
|
|
|
|
|
func CopySliceTo[F, T any](fromValue []F) []T {
|
|
|
|
|
|
var to []T
|
|
|
|
|
|
Copy(&to, fromValue)
|
|
|
|
|
|
return to
|
2020-09-01 10:34:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2021-01-08 15:37:32 +08:00
|
|
|
|
// 对结构体的每个字段以及字段值执行doWith回调函数, 包括匿名属性的字段
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func DoWithFields(str any, doWith func(fType reflect.StructField, fValue reflect.Value) error) error {
|
2021-01-08 15:37:32 +08:00
|
|
|
|
t := IndirectType(reflect.TypeOf(str))
|
|
|
|
|
|
if t.Kind() != reflect.Struct {
|
|
|
|
|
|
return errors.New("非结构体")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fieldNum := t.NumField()
|
|
|
|
|
|
v := Indirect(reflect.ValueOf(str))
|
|
|
|
|
|
for i := 0; i < fieldNum; i++ {
|
|
|
|
|
|
ft := t.Field(i)
|
|
|
|
|
|
fv := v.Field(i)
|
|
|
|
|
|
// 如果是匿名属性,则递归调用该方法
|
|
|
|
|
|
if ft.Anonymous {
|
|
|
|
|
|
DoWithFields(fv.Interface(), doWith)
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
err := doWith(ft, fv)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-09-01 10:34:11 +08:00
|
|
|
|
func Indirect(reflectValue reflect.Value) reflect.Value {
|
|
|
|
|
|
for reflectValue.Kind() == reflect.Ptr {
|
|
|
|
|
|
reflectValue = reflectValue.Elem()
|
|
|
|
|
|
}
|
|
|
|
|
|
return reflectValue
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func IndirectType(reflectType reflect.Type) reflect.Type {
|
|
|
|
|
|
for reflectType.Kind() == reflect.Ptr || reflectType.Kind() == reflect.Slice {
|
|
|
|
|
|
reflectType = reflectType.Elem()
|
|
|
|
|
|
}
|
|
|
|
|
|
return reflectType
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func Map2Struct(m map[string]any, s any) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
toValue := Indirect(reflect.ValueOf(s))
|
|
|
|
|
|
if !toValue.CanAddr() {
|
|
|
|
|
|
return errors.New("to value is unaddressable")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
innerStructMaps := getInnerStructMaps(m)
|
|
|
|
|
|
if len(innerStructMaps) != 0 {
|
|
|
|
|
|
for k, v := range innerStructMaps {
|
|
|
|
|
|
var fieldV reflect.Value
|
|
|
|
|
|
if strings.Contains(k, ".") {
|
|
|
|
|
|
fieldV = getFiledValueByPath(k, toValue)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
fieldV = toValue.FieldByName(k)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if !fieldV.CanSet() || !fieldV.CanAddr() {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
fieldT := fieldV.Type().Elem()
|
|
|
|
|
|
if fieldT.Kind() != reflect.Struct {
|
|
|
|
|
|
return errors.New(k + "不是结构体")
|
|
|
|
|
|
}
|
|
|
|
|
|
// 如果值为nil,则默认创建一个并赋值
|
|
|
|
|
|
if fieldV.IsNil() {
|
|
|
|
|
|
fieldV.Set(reflect.New(fieldT))
|
|
|
|
|
|
}
|
|
|
|
|
|
err := Map2Struct(v, fieldV.Addr().Interface())
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
var err error
|
|
|
|
|
|
for k, v := range m {
|
|
|
|
|
|
if v == nil {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
k = strings.Title(k)
|
|
|
|
|
|
// 如果key含有下划线,则将其转为驼峰
|
|
|
|
|
|
if strings.Contains(k, "_") {
|
|
|
|
|
|
k = Case2Camel(k)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fieldV := toValue.FieldByName(k)
|
|
|
|
|
|
if !fieldV.CanSet() {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
err = decode(k, v, fieldV)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func Maps2Structs(maps []map[string]any, structs any) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
structsV := reflect.Indirect(reflect.ValueOf(structs))
|
|
|
|
|
|
valType := structsV.Type()
|
|
|
|
|
|
valElemType := valType.Elem()
|
|
|
|
|
|
sliceType := reflect.SliceOf(valElemType)
|
|
|
|
|
|
|
|
|
|
|
|
length := len(maps)
|
|
|
|
|
|
|
|
|
|
|
|
valSlice := structsV
|
|
|
|
|
|
if valSlice.IsNil() {
|
|
|
|
|
|
// Make a new slice to hold our result, same size as the original data.
|
|
|
|
|
|
valSlice = reflect.MakeSlice(sliceType, length, length)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for i := 0; i < length; i++ {
|
|
|
|
|
|
err := Map2Struct(maps[i], valSlice.Index(i).Addr().Interface())
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
structsV.Set(valSlice)
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func getFiledValueByPath(path string, value reflect.Value) reflect.Value {
|
|
|
|
|
|
split := strings.Split(path, ".")
|
|
|
|
|
|
for _, v := range split {
|
|
|
|
|
|
if value.Type().Kind() == reflect.Ptr {
|
|
|
|
|
|
// 如果值为nil,则创建并赋值
|
|
|
|
|
|
if value.IsNil() {
|
|
|
|
|
|
value.Set(reflect.New(IndirectType(value.Type())))
|
|
|
|
|
|
}
|
|
|
|
|
|
value = value.Elem()
|
|
|
|
|
|
}
|
|
|
|
|
|
value = value.FieldByName(v)
|
|
|
|
|
|
}
|
|
|
|
|
|
return value
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func getInnerStructMaps(m map[string]any) map[string]map[string]any {
|
|
|
|
|
|
key2map := make(map[string]map[string]any)
|
2020-09-01 10:34:11 +08:00
|
|
|
|
for k, v := range m {
|
|
|
|
|
|
if !strings.Contains(k, ".") {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
lastIndex := strings.LastIndex(k, ".")
|
|
|
|
|
|
prefix := k[0:lastIndex]
|
|
|
|
|
|
m2 := key2map[prefix]
|
|
|
|
|
|
if m2 == nil {
|
2023-10-12 12:14:56 +08:00
|
|
|
|
key2map[prefix] = collx.M{k[lastIndex+1:]: v}
|
2020-09-01 10:34:11 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
m2[k[lastIndex+1:]] = v
|
|
|
|
|
|
}
|
|
|
|
|
|
delete(m, k)
|
|
|
|
|
|
}
|
|
|
|
|
|
return key2map
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// decode等方法摘抄自mapstructure库
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decode(name string, input any, outVal reflect.Value) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
var inputVal reflect.Value
|
|
|
|
|
|
if input != nil {
|
|
|
|
|
|
inputVal = reflect.ValueOf(input)
|
|
|
|
|
|
|
|
|
|
|
|
// We need to check here if input is a typed nil. Typed nils won't
|
|
|
|
|
|
// match the "input == nil" below so we check that here.
|
|
|
|
|
|
if inputVal.Kind() == reflect.Ptr && inputVal.IsNil() {
|
|
|
|
|
|
input = nil
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if !inputVal.IsValid() {
|
|
|
|
|
|
// If the input value is invalid, then we just set the value
|
|
|
|
|
|
// to be the zero value.
|
|
|
|
|
|
outVal.Set(reflect.Zero(outVal.Type()))
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var err error
|
|
|
|
|
|
outputKind := getKind(outVal)
|
|
|
|
|
|
switch outputKind {
|
|
|
|
|
|
case reflect.Int:
|
|
|
|
|
|
err = decodeInt(name, input, outVal)
|
|
|
|
|
|
case reflect.Uint:
|
|
|
|
|
|
err = decodeUint(name, input, outVal)
|
|
|
|
|
|
case reflect.Float32:
|
|
|
|
|
|
err = decodeFloat(name, input, outVal)
|
|
|
|
|
|
case reflect.String:
|
|
|
|
|
|
err = decodeString(name, input, outVal)
|
|
|
|
|
|
case reflect.Ptr:
|
|
|
|
|
|
_, err = decodePtr(name, input, outVal)
|
|
|
|
|
|
default:
|
|
|
|
|
|
// If we reached this point then we weren't able to decode it
|
|
|
|
|
|
return fmt.Errorf("%s: unsupported type: %s", name, outputKind)
|
|
|
|
|
|
}
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decodeInt(name string, data any, val reflect.Value) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
dataVal := reflect.Indirect(reflect.ValueOf(data))
|
|
|
|
|
|
dataKind := getKind(dataVal)
|
|
|
|
|
|
dataType := dataVal.Type()
|
|
|
|
|
|
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case dataKind == reflect.Int:
|
|
|
|
|
|
val.SetInt(dataVal.Int())
|
|
|
|
|
|
case dataKind == reflect.Uint:
|
|
|
|
|
|
val.SetInt(int64(dataVal.Uint()))
|
|
|
|
|
|
case dataKind == reflect.Float32:
|
|
|
|
|
|
val.SetInt(int64(dataVal.Float()))
|
|
|
|
|
|
case dataKind == reflect.Bool:
|
|
|
|
|
|
if dataVal.Bool() {
|
|
|
|
|
|
val.SetInt(1)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
val.SetInt(0)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataKind == reflect.String:
|
|
|
|
|
|
i, err := strconv.ParseInt(dataVal.String(), 0, val.Type().Bits())
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
val.SetInt(i)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s' as int: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataType.PkgPath() == "encoding/json" && dataType.Name() == "Number":
|
|
|
|
|
|
jn := data.(json.Number)
|
|
|
|
|
|
i, err := jn.Int64()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"error decoding json.Number into %s: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetInt(i)
|
|
|
|
|
|
default:
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"'%s' expected type '%s', got unconvertible type '%s'",
|
|
|
|
|
|
name, val.Type(), dataVal.Type())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decodeUint(name string, data any, val reflect.Value) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
dataVal := reflect.Indirect(reflect.ValueOf(data))
|
|
|
|
|
|
dataKind := getKind(dataVal)
|
|
|
|
|
|
dataType := dataVal.Type()
|
|
|
|
|
|
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case dataKind == reflect.Int:
|
|
|
|
|
|
i := dataVal.Int()
|
|
|
|
|
|
if i < 0 {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s', %d overflows uint",
|
|
|
|
|
|
name, i)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetUint(uint64(i))
|
|
|
|
|
|
case dataKind == reflect.Uint:
|
|
|
|
|
|
val.SetUint(dataVal.Uint())
|
|
|
|
|
|
case dataKind == reflect.Float32:
|
|
|
|
|
|
f := dataVal.Float()
|
|
|
|
|
|
if f < 0 {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s', %f overflows uint",
|
|
|
|
|
|
name, f)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetUint(uint64(f))
|
|
|
|
|
|
case dataKind == reflect.Bool:
|
|
|
|
|
|
if dataVal.Bool() {
|
|
|
|
|
|
val.SetUint(1)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
val.SetUint(0)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataKind == reflect.String:
|
|
|
|
|
|
i, err := strconv.ParseUint(dataVal.String(), 0, val.Type().Bits())
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
val.SetUint(i)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s' as uint: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataType.PkgPath() == "encoding/json" && dataType.Name() == "Number":
|
|
|
|
|
|
jn := data.(json.Number)
|
|
|
|
|
|
i, err := jn.Int64()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"error decoding json.Number into %s: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
if i < 0 {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s', %d overflows uint",
|
|
|
|
|
|
name, i)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetUint(uint64(i))
|
|
|
|
|
|
default:
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"'%s' expected type '%s', got unconvertible type '%s'",
|
|
|
|
|
|
name, val.Type(), dataVal.Type())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decodeFloat(name string, data any, val reflect.Value) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
dataVal := reflect.Indirect(reflect.ValueOf(data))
|
|
|
|
|
|
dataKind := getKind(dataVal)
|
|
|
|
|
|
dataType := dataVal.Type()
|
|
|
|
|
|
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case dataKind == reflect.Int:
|
|
|
|
|
|
val.SetFloat(float64(dataVal.Int()))
|
|
|
|
|
|
case dataKind == reflect.Uint:
|
|
|
|
|
|
val.SetFloat(float64(dataVal.Uint()))
|
|
|
|
|
|
case dataKind == reflect.Float32:
|
|
|
|
|
|
val.SetFloat(dataVal.Float())
|
|
|
|
|
|
case dataKind == reflect.Bool:
|
|
|
|
|
|
if dataVal.Bool() {
|
|
|
|
|
|
val.SetFloat(1)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
val.SetFloat(0)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataKind == reflect.String:
|
|
|
|
|
|
f, err := strconv.ParseFloat(dataVal.String(), val.Type().Bits())
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
val.SetFloat(f)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return fmt.Errorf("cannot parse '%s' as float: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataType.PkgPath() == "encoding/json" && dataType.Name() == "Number":
|
|
|
|
|
|
jn := data.(json.Number)
|
|
|
|
|
|
i, err := jn.Float64()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"error decoding json.Number into %s: %s", name, err)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetFloat(i)
|
|
|
|
|
|
default:
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"'%s' expected type '%s', got unconvertible type '%s'",
|
|
|
|
|
|
name, val.Type(), dataVal.Type())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decodeString(name string, data any, val reflect.Value) error {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
dataVal := reflect.Indirect(reflect.ValueOf(data))
|
|
|
|
|
|
dataKind := getKind(dataVal)
|
|
|
|
|
|
|
|
|
|
|
|
converted := true
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case dataKind == reflect.String:
|
|
|
|
|
|
val.SetString(dataVal.String())
|
|
|
|
|
|
case dataKind == reflect.Bool:
|
|
|
|
|
|
if dataVal.Bool() {
|
|
|
|
|
|
val.SetString("1")
|
|
|
|
|
|
} else {
|
|
|
|
|
|
val.SetString("0")
|
|
|
|
|
|
}
|
|
|
|
|
|
case dataKind == reflect.Int:
|
|
|
|
|
|
val.SetString(strconv.FormatInt(dataVal.Int(), 10))
|
|
|
|
|
|
case dataKind == reflect.Uint:
|
|
|
|
|
|
val.SetString(strconv.FormatUint(dataVal.Uint(), 10))
|
|
|
|
|
|
case dataKind == reflect.Float32:
|
|
|
|
|
|
val.SetString(strconv.FormatFloat(dataVal.Float(), 'f', -1, 64))
|
|
|
|
|
|
case dataKind == reflect.Slice,
|
|
|
|
|
|
dataKind == reflect.Array:
|
|
|
|
|
|
dataType := dataVal.Type()
|
|
|
|
|
|
elemKind := dataType.Elem().Kind()
|
|
|
|
|
|
switch elemKind {
|
|
|
|
|
|
case reflect.Uint8:
|
|
|
|
|
|
var uints []uint8
|
|
|
|
|
|
if dataKind == reflect.Array {
|
|
|
|
|
|
uints = make([]uint8, dataVal.Len(), dataVal.Len())
|
|
|
|
|
|
for i := range uints {
|
|
|
|
|
|
uints[i] = dataVal.Index(i).Interface().(uint8)
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
uints = dataVal.Interface().([]uint8)
|
|
|
|
|
|
}
|
|
|
|
|
|
val.SetString(string(uints))
|
|
|
|
|
|
default:
|
|
|
|
|
|
converted = false
|
|
|
|
|
|
}
|
|
|
|
|
|
default:
|
|
|
|
|
|
converted = false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if !converted {
|
|
|
|
|
|
return fmt.Errorf(
|
|
|
|
|
|
"'%s' expected type '%s', got unconvertible type '%s'",
|
|
|
|
|
|
name, val.Type(), dataVal.Type())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-06-01 12:31:32 +08:00
|
|
|
|
func decodePtr(name string, data any, val reflect.Value) (bool, error) {
|
2020-09-01 10:34:11 +08:00
|
|
|
|
// If the input data is nil, then we want to just set the output
|
|
|
|
|
|
// pointer to be nil as well.
|
|
|
|
|
|
isNil := data == nil
|
|
|
|
|
|
if !isNil {
|
|
|
|
|
|
switch v := reflect.Indirect(reflect.ValueOf(data)); v.Kind() {
|
|
|
|
|
|
case reflect.Chan,
|
|
|
|
|
|
reflect.Func,
|
|
|
|
|
|
reflect.Interface,
|
|
|
|
|
|
reflect.Map,
|
|
|
|
|
|
reflect.Ptr,
|
|
|
|
|
|
reflect.Slice:
|
|
|
|
|
|
isNil = v.IsNil()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if isNil {
|
|
|
|
|
|
if !val.IsNil() && val.CanSet() {
|
|
|
|
|
|
nilValue := reflect.New(val.Type()).Elem()
|
|
|
|
|
|
val.Set(nilValue)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Create an element of the concrete (non pointer) type and decode
|
|
|
|
|
|
// into that. Then set the value of the pointer to this type.
|
|
|
|
|
|
valType := val.Type()
|
|
|
|
|
|
valElemType := valType.Elem()
|
|
|
|
|
|
if val.CanSet() {
|
|
|
|
|
|
realVal := val
|
|
|
|
|
|
if realVal.IsNil() {
|
|
|
|
|
|
realVal = reflect.New(valElemType)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if err := decode(name, data, reflect.Indirect(realVal)); err != nil {
|
|
|
|
|
|
return false, err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
val.Set(realVal)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if err := decode(name, data, reflect.Indirect(val)); err != nil {
|
|
|
|
|
|
return false, err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return false, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func getKind(val reflect.Value) reflect.Kind {
|
|
|
|
|
|
kind := val.Kind()
|
|
|
|
|
|
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case kind >= reflect.Int && kind <= reflect.Int64:
|
|
|
|
|
|
return reflect.Int
|
|
|
|
|
|
case kind >= reflect.Uint && kind <= reflect.Uint64:
|
|
|
|
|
|
return reflect.Uint
|
|
|
|
|
|
case kind >= reflect.Float32 && kind <= reflect.Float64:
|
|
|
|
|
|
return reflect.Float32
|
|
|
|
|
|
default:
|
|
|
|
|
|
return kind
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 下划线写法转为驼峰写法
|
|
|
|
|
|
func Case2Camel(name string) string {
|
|
|
|
|
|
name = strings.Replace(name, "_", " ", -1)
|
|
|
|
|
|
name = strings.Title(name)
|
|
|
|
|
|
return strings.Replace(name, " ", "", -1)
|
|
|
|
|
|
}
|
2024-02-29 22:12:50 +08:00
|
|
|
|
|
|
|
|
|
|
// 结构体转为map
|
|
|
|
|
|
func ToMap(input any) map[string]any {
|
|
|
|
|
|
result := make(map[string]any)
|
|
|
|
|
|
v := Indirect(reflect.ValueOf(input))
|
|
|
|
|
|
if v.Kind() != reflect.Struct {
|
|
|
|
|
|
return result
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for i := 0; i < v.NumField(); i++ {
|
|
|
|
|
|
field := v.Field(i)
|
|
|
|
|
|
zeroValue := reflect.Zero(field.Type()).Interface()
|
|
|
|
|
|
fieldValue := field.Interface()
|
|
|
|
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(fieldValue, zeroValue) {
|
|
|
|
|
|
result[v.Type().Field(i).Name] = fieldValue
|
|
|
|
|
|
} else {
|
|
|
|
|
|
result[v.Type().Field(i).Name] = zeroValue
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
}
|