mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 11:18:50 +08:00
347 lines
12 KiB
Go
347 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 gconv
|
|
|
|
import (
|
|
"reflect"
|
|
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/internal/json"
|
|
"github.com/gogf/gf/v2/util/gconv/internal/localinterface"
|
|
)
|
|
|
|
// Scan automatically checks the type of `pointer` and converts `params` to `pointer`.
|
|
// It supports various types of parameter conversions, including:
|
|
// 1. Basic types (int, string, float, etc.)
|
|
// 2. Pointer types
|
|
// 3. Slice types
|
|
// 4. Map types
|
|
// 5. Struct types
|
|
//
|
|
// The `paramKeyToAttrMap` parameter is used for mapping between attribute names and parameter keys.
|
|
// TODO: change `paramKeyToAttrMap` to `ScanOption` to be more scalable; add `DeepCopy` option for `ScanOption`.
|
|
func Scan(srcValue any, dstPointer any, paramKeyToAttrMap ...map[string]string) (err error) {
|
|
// Check if srcValue is nil, in which case no conversion is needed
|
|
if srcValue == nil {
|
|
return nil
|
|
}
|
|
// Check if dstPointer is nil, which is an invalid parameter
|
|
if dstPointer == nil {
|
|
return gerror.NewCode(
|
|
gcode.CodeInvalidParameter,
|
|
`destination pointer should not be nil`,
|
|
)
|
|
}
|
|
|
|
// Get the reflection type and value of dstPointer
|
|
var (
|
|
dstPointerReflectType reflect.Type
|
|
dstPointerReflectValue reflect.Value
|
|
)
|
|
if v, ok := dstPointer.(reflect.Value); ok {
|
|
dstPointerReflectValue = v
|
|
dstPointerReflectType = v.Type()
|
|
} else {
|
|
dstPointerReflectValue = reflect.ValueOf(dstPointer)
|
|
// Do not use dstPointerReflectValue.Type() as dstPointerReflectValue might be zero
|
|
dstPointerReflectType = reflect.TypeOf(dstPointer)
|
|
}
|
|
|
|
// Validate the kind of dstPointer
|
|
var dstPointerReflectKind = dstPointerReflectType.Kind()
|
|
if dstPointerReflectKind != reflect.Ptr {
|
|
// If dstPointer is not a pointer, try to get its address
|
|
if dstPointerReflectValue.CanAddr() {
|
|
dstPointerReflectValue = dstPointerReflectValue.Addr()
|
|
dstPointerReflectType = dstPointerReflectValue.Type()
|
|
dstPointerReflectKind = dstPointerReflectType.Kind()
|
|
} else {
|
|
// If dstPointer is not a pointer and cannot be addressed, return an error
|
|
return gerror.NewCodef(
|
|
gcode.CodeInvalidParameter,
|
|
`destination pointer should be type of pointer, but got type: %v`,
|
|
dstPointerReflectType,
|
|
)
|
|
}
|
|
}
|
|
|
|
// Get the reflection value of srcValue
|
|
var srcValueReflectValue reflect.Value
|
|
if v, ok := srcValue.(reflect.Value); ok {
|
|
srcValueReflectValue = v
|
|
} else {
|
|
srcValueReflectValue = reflect.ValueOf(srcValue)
|
|
}
|
|
|
|
// Get the element type and kind of dstPointer
|
|
var (
|
|
dstPointerReflectValueElem = dstPointerReflectValue.Elem()
|
|
dstPointerReflectValueElemKind = dstPointerReflectValueElem.Kind()
|
|
)
|
|
// Handle multiple level pointers
|
|
if dstPointerReflectValueElemKind == reflect.Ptr {
|
|
if dstPointerReflectValueElem.IsNil() {
|
|
// Create a new value for the pointer dereference
|
|
nextLevelPtr := reflect.New(dstPointerReflectValueElem.Type().Elem())
|
|
// Recursively scan into the dereferenced pointer
|
|
if err = Scan(srcValueReflectValue, nextLevelPtr, paramKeyToAttrMap...); err == nil {
|
|
dstPointerReflectValueElem.Set(nextLevelPtr)
|
|
}
|
|
return
|
|
}
|
|
return Scan(srcValueReflectValue, dstPointerReflectValueElem, paramKeyToAttrMap...)
|
|
}
|
|
|
|
// Check if srcValue and dstPointer are the same type, in which case direct assignment can be performed
|
|
if ok := doConvertWithTypeCheck(srcValueReflectValue, dstPointerReflectValueElem); ok {
|
|
return nil
|
|
}
|
|
|
|
// Handle different destination types
|
|
switch dstPointerReflectValueElemKind {
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
// Convert to int type
|
|
dstPointerReflectValueElem.SetInt(Int64(srcValue))
|
|
return nil
|
|
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
|
// Convert to uint type
|
|
dstPointerReflectValueElem.SetUint(Uint64(srcValue))
|
|
return nil
|
|
|
|
case reflect.Float32, reflect.Float64:
|
|
// Convert to float type
|
|
dstPointerReflectValueElem.SetFloat(Float64(srcValue))
|
|
return nil
|
|
|
|
case reflect.String:
|
|
// Convert to string type
|
|
dstPointerReflectValueElem.SetString(String(srcValue))
|
|
return nil
|
|
|
|
case reflect.Bool:
|
|
// Convert to bool type
|
|
dstPointerReflectValueElem.SetBool(Bool(srcValue))
|
|
return nil
|
|
|
|
case reflect.Slice:
|
|
// Handle slice type conversion
|
|
var (
|
|
dstElemType = dstPointerReflectValueElem.Type().Elem()
|
|
dstElemKind = dstElemType.Kind()
|
|
)
|
|
// The slice element might be a pointer type
|
|
if dstElemKind == reflect.Ptr {
|
|
dstElemType = dstElemType.Elem()
|
|
dstElemKind = dstElemType.Kind()
|
|
}
|
|
// Special handling for struct or map slice elements
|
|
if dstElemKind == reflect.Struct || dstElemKind == reflect.Map {
|
|
return doScanForComplicatedTypes(srcValue, dstPointer, dstPointerReflectType, paramKeyToAttrMap...)
|
|
}
|
|
// Handle basic type slice conversions
|
|
var srcValueReflectValueKind = srcValueReflectValue.Kind()
|
|
if srcValueReflectValueKind == reflect.Slice || srcValueReflectValueKind == reflect.Array {
|
|
var (
|
|
srcLen = srcValueReflectValue.Len()
|
|
newSlice = reflect.MakeSlice(dstPointerReflectValueElem.Type(), srcLen, srcLen)
|
|
)
|
|
for i := 0; i < srcLen; i++ {
|
|
srcElem := srcValueReflectValue.Index(i).Interface()
|
|
switch dstElemType.Kind() {
|
|
case reflect.String:
|
|
newSlice.Index(i).SetString(String(srcElem))
|
|
case reflect.Int:
|
|
newSlice.Index(i).SetInt(Int64(srcElem))
|
|
case reflect.Int64:
|
|
newSlice.Index(i).SetInt(Int64(srcElem))
|
|
case reflect.Float64:
|
|
newSlice.Index(i).SetFloat(Float64(srcElem))
|
|
case reflect.Bool:
|
|
newSlice.Index(i).SetBool(Bool(srcElem))
|
|
default:
|
|
return Scan(
|
|
srcElem, newSlice.Index(i).Addr().Interface(), paramKeyToAttrMap...,
|
|
)
|
|
}
|
|
}
|
|
dstPointerReflectValueElem.Set(newSlice)
|
|
return nil
|
|
}
|
|
return doScanForComplicatedTypes(srcValue, dstPointer, dstPointerReflectType, paramKeyToAttrMap...)
|
|
|
|
default:
|
|
// Handle complex types (structs, maps, etc.)
|
|
return doScanForComplicatedTypes(srcValue, dstPointer, dstPointerReflectType, paramKeyToAttrMap...)
|
|
}
|
|
}
|
|
|
|
// doScanForComplicatedTypes handles the scanning of complex data types.
|
|
// It supports converting between maps, structs, and slices of these types.
|
|
// The function first attempts JSON conversion, then falls back to specific type handling.
|
|
//
|
|
// It supports `pointer` in type of `*map/*[]map/*[]*map/*struct/**struct/*[]struct/*[]*struct` for converting.
|
|
//
|
|
// Parameters:
|
|
// - srcValue: The source value to convert from
|
|
// - dstPointer: The destination pointer to convert to
|
|
// - dstPointerReflectType: The reflection type of the destination pointer
|
|
// - paramKeyToAttrMap: Optional mapping between parameter keys and struct attribute names
|
|
func doScanForComplicatedTypes(
|
|
srcValue, dstPointer any,
|
|
dstPointerReflectType reflect.Type,
|
|
paramKeyToAttrMap ...map[string]string,
|
|
) error {
|
|
// Try JSON conversion first
|
|
ok, err := doConvertWithJsonCheck(srcValue, dstPointer)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if ok {
|
|
return nil
|
|
}
|
|
|
|
// Handle specific type conversions
|
|
var (
|
|
dstPointerReflectTypeElem = dstPointerReflectType.Elem()
|
|
dstPointerReflectTypeElemKind = dstPointerReflectTypeElem.Kind()
|
|
keyToAttributeNameMapping map[string]string
|
|
)
|
|
if len(paramKeyToAttrMap) > 0 {
|
|
keyToAttributeNameMapping = paramKeyToAttrMap[0]
|
|
}
|
|
|
|
// Handle different destination types
|
|
switch dstPointerReflectTypeElemKind {
|
|
case reflect.Map:
|
|
// Convert map to map
|
|
return doMapToMap(srcValue, dstPointer, paramKeyToAttrMap...)
|
|
|
|
case reflect.Array, reflect.Slice:
|
|
var (
|
|
sliceElem = dstPointerReflectTypeElem.Elem()
|
|
sliceElemKind = sliceElem.Kind()
|
|
)
|
|
// Handle pointer elements
|
|
for sliceElemKind == reflect.Ptr {
|
|
sliceElem = sliceElem.Elem()
|
|
sliceElemKind = sliceElem.Kind()
|
|
}
|
|
if sliceElemKind == reflect.Map {
|
|
// Convert to slice of maps
|
|
return doMapToMaps(srcValue, dstPointer, paramKeyToAttrMap...)
|
|
}
|
|
// Convert to slice of structs
|
|
return doStructs(srcValue, dstPointer, keyToAttributeNameMapping, "")
|
|
|
|
default:
|
|
// Convert to single struct
|
|
return doStruct(srcValue, dstPointer, keyToAttributeNameMapping, "")
|
|
}
|
|
}
|
|
|
|
// doConvertWithTypeCheck supports `pointer` in type of `*map/*[]map/*[]*map/*struct/**struct/*[]struct/*[]*struct`
|
|
// for converting.
|
|
func doConvertWithTypeCheck(srcValueReflectValue, dstPointerReflectValueElem reflect.Value) (ok bool) {
|
|
if !dstPointerReflectValueElem.IsValid() || !srcValueReflectValue.IsValid() {
|
|
return false
|
|
}
|
|
switch {
|
|
// Examples:
|
|
// UploadFile => UploadFile
|
|
// []UploadFile => []UploadFile
|
|
// *UploadFile => *UploadFile
|
|
// *[]UploadFile => *[]UploadFile
|
|
// map[int][int] => map[int][int]
|
|
// []map[int][int] => []map[int][int]
|
|
// *[]map[int][int] => *[]map[int][int]
|
|
case dstPointerReflectValueElem.Type() == srcValueReflectValue.Type():
|
|
dstPointerReflectValueElem.Set(srcValueReflectValue)
|
|
return true
|
|
|
|
// Examples:
|
|
// UploadFile => *UploadFile
|
|
// []UploadFile => *[]UploadFile
|
|
// map[int][int] => *map[int][int]
|
|
// []map[int][int] => *[]map[int][int]
|
|
case dstPointerReflectValueElem.Kind() == reflect.Ptr &&
|
|
dstPointerReflectValueElem.Elem().IsValid() &&
|
|
dstPointerReflectValueElem.Elem().Type() == srcValueReflectValue.Type():
|
|
dstPointerReflectValueElem.Elem().Set(srcValueReflectValue)
|
|
return true
|
|
|
|
// Examples:
|
|
// *UploadFile => UploadFile
|
|
// *[]UploadFile => []UploadFile
|
|
// *map[int][int] => map[int][int]
|
|
// *[]map[int][int] => []map[int][int]
|
|
case srcValueReflectValue.Kind() == reflect.Ptr &&
|
|
srcValueReflectValue.Elem().IsValid() &&
|
|
dstPointerReflectValueElem.Type() == srcValueReflectValue.Elem().Type():
|
|
dstPointerReflectValueElem.Set(srcValueReflectValue.Elem())
|
|
return true
|
|
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// doConvertWithJsonCheck attempts to convert the source value to the destination
|
|
// using JSON marshaling and unmarshaling. This is particularly useful for complex
|
|
// types that can be represented as JSON.
|
|
//
|
|
// Parameters:
|
|
// - srcValue: The source value to convert from
|
|
// - dstPointer: The destination pointer to convert to
|
|
//
|
|
// Returns:
|
|
// - bool: true if JSON conversion was successful
|
|
// - error: any error that occurred during conversion
|
|
func doConvertWithJsonCheck(srcValue any, dstPointer any) (ok bool, err error) {
|
|
switch valueResult := srcValue.(type) {
|
|
case []byte:
|
|
if json.Valid(valueResult) {
|
|
if dstPointerReflectType, ok := dstPointer.(reflect.Value); ok {
|
|
if dstPointerReflectType.Kind() == reflect.Ptr {
|
|
if dstPointerReflectType.IsNil() {
|
|
return false, nil
|
|
}
|
|
return true, json.UnmarshalUseNumber(valueResult, dstPointerReflectType.Interface())
|
|
} else if dstPointerReflectType.CanAddr() {
|
|
return true, json.UnmarshalUseNumber(valueResult, dstPointerReflectType.Addr().Interface())
|
|
}
|
|
} else {
|
|
return true, json.UnmarshalUseNumber(valueResult, dstPointer)
|
|
}
|
|
}
|
|
|
|
case string:
|
|
if valueBytes := []byte(valueResult); json.Valid(valueBytes) {
|
|
if dstPointerReflectType, ok := dstPointer.(reflect.Value); ok {
|
|
if dstPointerReflectType.Kind() == reflect.Ptr {
|
|
if dstPointerReflectType.IsNil() {
|
|
return false, nil
|
|
}
|
|
return true, json.UnmarshalUseNumber(valueBytes, dstPointerReflectType.Interface())
|
|
} else if dstPointerReflectType.CanAddr() {
|
|
return true, json.UnmarshalUseNumber(valueBytes, dstPointerReflectType.Addr().Interface())
|
|
}
|
|
} else {
|
|
return true, json.UnmarshalUseNumber(valueBytes, dstPointer)
|
|
}
|
|
}
|
|
|
|
default:
|
|
// The `params` might be struct that implements interface function Interface, eg: gvar.Var.
|
|
if v, ok := srcValue.(localinterface.IInterface); ok {
|
|
return doConvertWithJsonCheck(v.Interface(), dstPointer)
|
|
}
|
|
}
|
|
return false, nil
|
|
}
|