188 lines
3.7 KiB
Go
Executable File
188 lines
3.7 KiB
Go
Executable File
package mapping
|
||
|
||
import (
|
||
"fmt"
|
||
"reflect"
|
||
"strings"
|
||
)
|
||
|
||
const (
|
||
emptyTag = ""
|
||
tagKVSeparator = ":"
|
||
)
|
||
|
||
// Marshal marshals the given val and returns the map that contains the fields.
|
||
// optional=another is not implemented, and it's hard to implement and not commonly used.
|
||
func Marshal(val any) (map[string]map[string]any, error) {
|
||
ret := make(map[string]map[string]any)
|
||
tp := reflect.TypeOf(val)
|
||
if tp.Kind() == reflect.Ptr {
|
||
tp = tp.Elem()
|
||
}
|
||
rv := reflect.ValueOf(val)
|
||
if rv.Kind() == reflect.Ptr {
|
||
rv = rv.Elem()
|
||
}
|
||
|
||
for i := 0; i < tp.NumField(); i++ {
|
||
field := tp.Field(i)
|
||
value := rv.Field(i)
|
||
if err := processMember(field, value, ret); err != nil {
|
||
return nil, err
|
||
}
|
||
}
|
||
|
||
return ret, nil
|
||
}
|
||
|
||
func getTag(field reflect.StructField) (string, bool) {
|
||
tag := string(field.Tag)
|
||
if i := strings.Index(tag, tagKVSeparator); i >= 0 {
|
||
return strings.TrimSpace(tag[:i]), true
|
||
}
|
||
|
||
return strings.TrimSpace(tag), false
|
||
}
|
||
|
||
func processMember(field reflect.StructField, value reflect.Value,
|
||
collector map[string]map[string]any) error {
|
||
var key string
|
||
var opt *fieldOptions
|
||
var err error
|
||
tag, ok := getTag(field)
|
||
if !ok {
|
||
tag = emptyTag
|
||
key = field.Name
|
||
} else {
|
||
key, opt, err = parseKeyAndOptions(tag, field)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
if err = validate(field, value, opt); err != nil {
|
||
return err
|
||
}
|
||
}
|
||
|
||
val := value.Interface()
|
||
if opt != nil && opt.FromString {
|
||
val = fmt.Sprint(val)
|
||
}
|
||
|
||
m, ok := collector[tag]
|
||
if ok {
|
||
m[key] = val
|
||
} else {
|
||
m = map[string]any{
|
||
key: val,
|
||
}
|
||
}
|
||
collector[tag] = m
|
||
|
||
return nil
|
||
}
|
||
|
||
func validate(field reflect.StructField, value reflect.Value, opt *fieldOptions) error {
|
||
// 根据开发使用习惯,屏蔽因未配置optional而导致序列化失败的问题
|
||
//if opt == nil || !opt.Optional {
|
||
// if err := validateOptional(field, value); err != nil {
|
||
// return err
|
||
// }
|
||
//}
|
||
|
||
if opt == nil {
|
||
return nil
|
||
}
|
||
|
||
if opt.Optional && value.IsZero() {
|
||
return nil
|
||
}
|
||
|
||
if len(opt.Options) > 0 {
|
||
if err := validateOptions(value, opt); err != nil {
|
||
return err
|
||
}
|
||
}
|
||
|
||
if opt.Range != nil {
|
||
if err := validateRange(value, opt); err != nil {
|
||
return err
|
||
}
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
func validateOptional(field reflect.StructField, value reflect.Value) error {
|
||
switch field.Type.Kind() {
|
||
case reflect.Ptr:
|
||
if value.IsNil() {
|
||
return fmt.Errorf("field %q is nil", field.Name)
|
||
}
|
||
case reflect.Array, reflect.Slice, reflect.Map:
|
||
if value.IsNil() || value.Len() == 0 {
|
||
return fmt.Errorf("field %q is empty", field.Name)
|
||
}
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
func validateOptions(value reflect.Value, opt *fieldOptions) error {
|
||
var found bool
|
||
val := fmt.Sprint(value.Interface())
|
||
for i := range opt.Options {
|
||
if opt.Options[i] == val {
|
||
found = true
|
||
break
|
||
}
|
||
}
|
||
if !found {
|
||
return fmt.Errorf("field %q not in options", val)
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
func validateRange(value reflect.Value, opt *fieldOptions) error {
|
||
var val float64
|
||
switch v := value.Interface().(type) {
|
||
case int:
|
||
val = float64(v)
|
||
case int8:
|
||
val = float64(v)
|
||
case int16:
|
||
val = float64(v)
|
||
case int32:
|
||
val = float64(v)
|
||
case int64:
|
||
val = float64(v)
|
||
case uint:
|
||
val = float64(v)
|
||
case uint8:
|
||
val = float64(v)
|
||
case uint16:
|
||
val = float64(v)
|
||
case uint32:
|
||
val = float64(v)
|
||
case uint64:
|
||
val = float64(v)
|
||
case float32:
|
||
val = float64(v)
|
||
case float64:
|
||
val = v
|
||
default:
|
||
return fmt.Errorf("unknown support type for range %q", value.Type().String())
|
||
}
|
||
|
||
// validates [left, right], [left, right), (left, right], (left, right)
|
||
if val < opt.Range.left ||
|
||
(!opt.Range.leftInclude && val == opt.Range.left) ||
|
||
val > opt.Range.right ||
|
||
(!opt.Range.rightInclude && val == opt.Range.right) {
|
||
return fmt.Errorf("%v out of range", value.Interface())
|
||
}
|
||
|
||
return nil
|
||
}
|