mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
142 lines
4.0 KiB
Go
142 lines
4.0 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 converter
|
|
|
|
import (
|
|
"reflect"
|
|
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
)
|
|
|
|
// MapToMap converts any map type variable `params` to another map type variable `pointer`.
|
|
//
|
|
// The parameter `params` can be any type of map, like:
|
|
// map[string]string, map[string]struct, map[string]*struct, reflect.Value, etc.
|
|
//
|
|
// The parameter `pointer` should be type of *map, like:
|
|
// map[int]string, map[string]struct, map[string]*struct, reflect.Value, etc.
|
|
//
|
|
// The optional parameter `mapping` is used for struct attribute to map key mapping, which makes
|
|
// sense only if the items of original map `params` is type struct.
|
|
func (c *Converter) MapToMap(
|
|
params, pointer any, mapping map[string]string, option ...MapOption,
|
|
) (err error) {
|
|
var (
|
|
paramsRv reflect.Value
|
|
paramsKind reflect.Kind
|
|
)
|
|
if v, ok := params.(reflect.Value); ok {
|
|
paramsRv = v
|
|
} else {
|
|
paramsRv = reflect.ValueOf(params)
|
|
}
|
|
paramsKind = paramsRv.Kind()
|
|
if paramsKind == reflect.Ptr {
|
|
paramsRv = paramsRv.Elem()
|
|
paramsKind = paramsRv.Kind()
|
|
}
|
|
if paramsKind != reflect.Map {
|
|
m, err := c.Map(params, option...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.MapToMap(m, pointer, mapping, option...)
|
|
}
|
|
// Empty params map, no need continue.
|
|
if paramsRv.Len() == 0 {
|
|
return nil
|
|
}
|
|
var pointerRv reflect.Value
|
|
if v, ok := pointer.(reflect.Value); ok {
|
|
pointerRv = v
|
|
} else {
|
|
pointerRv = reflect.ValueOf(pointer)
|
|
}
|
|
pointerKind := pointerRv.Kind()
|
|
for pointerKind == reflect.Ptr {
|
|
pointerRv = pointerRv.Elem()
|
|
pointerKind = pointerRv.Kind()
|
|
}
|
|
if pointerKind != reflect.Map {
|
|
return gerror.NewCodef(
|
|
gcode.CodeInvalidParameter,
|
|
`destination pointer should be type of *map, but got: %s`,
|
|
pointerKind,
|
|
)
|
|
}
|
|
defer func() {
|
|
// Catch the panic, especially the reflection operation panics.
|
|
if exception := recover(); exception != nil {
|
|
if v, ok := exception.(error); ok && gerror.HasStack(v) {
|
|
err = v
|
|
} else {
|
|
err = gerror.NewCodeSkipf(gcode.CodeInternalPanic, 1, "%+v", exception)
|
|
}
|
|
}
|
|
}()
|
|
var (
|
|
paramsKeys = paramsRv.MapKeys()
|
|
pointerKeyType = pointerRv.Type().Key()
|
|
pointerValueType = pointerRv.Type().Elem()
|
|
pointerValueKind = pointerValueType.Kind()
|
|
dataMap = reflect.MakeMapWithSize(pointerRv.Type(), len(paramsKeys))
|
|
mapOption = c.getMapOption(option...)
|
|
convertOption = ConvertOption{
|
|
StructOption: StructOption{ContinueOnError: mapOption.ContinueOnError},
|
|
SliceOption: SliceOption{ContinueOnError: mapOption.ContinueOnError},
|
|
MapOption: mapOption,
|
|
}
|
|
)
|
|
// Retrieve the true element type of target map.
|
|
if pointerValueKind == reflect.Ptr {
|
|
pointerValueKind = pointerValueType.Elem().Kind()
|
|
}
|
|
for _, key := range paramsKeys {
|
|
mapValue := reflect.New(pointerValueType).Elem()
|
|
switch pointerValueKind {
|
|
case reflect.Map, reflect.Struct:
|
|
structOption := StructOption{
|
|
ParamKeyToAttrMap: mapping,
|
|
PriorityTag: "",
|
|
ContinueOnError: mapOption.ContinueOnError,
|
|
}
|
|
if err = c.Struct(paramsRv.MapIndex(key).Interface(), mapValue, structOption); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
convertResult, err := c.doConvert(
|
|
doConvertInput{
|
|
FromValue: paramsRv.MapIndex(key).Interface(),
|
|
ToTypeName: pointerValueType.String(),
|
|
ReferValue: mapValue,
|
|
},
|
|
convertOption,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
mapValue.Set(reflect.ValueOf(convertResult))
|
|
}
|
|
convertResult, err := c.doConvert(
|
|
doConvertInput{
|
|
FromValue: key.Interface(),
|
|
ToTypeName: pointerKeyType.Name(),
|
|
ReferValue: reflect.New(pointerKeyType).Elem().Interface(),
|
|
},
|
|
convertOption,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var mapKey = reflect.ValueOf(convertResult)
|
|
dataMap.SetMapIndex(mapKey, mapValue)
|
|
}
|
|
pointerRv.Set(dataMap)
|
|
return nil
|
|
}
|