mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
119 lines
4.0 KiB
Go
119 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 structcache
|
|
|
|
import (
|
|
"reflect"
|
|
|
|
"github.com/gogf/gf/v2/internal/utils"
|
|
"github.com/gogf/gf/v2/util/gtag"
|
|
)
|
|
|
|
// GetCachedStructInfo retrieves or parses and returns a cached info for certain struct type.
|
|
// The given `structType` should be type of struct.
|
|
func (cf *Converter) GetCachedStructInfo(structType reflect.Type, priorityTag string) *CachedStructInfo {
|
|
if structType.Kind() != reflect.Struct {
|
|
return nil
|
|
}
|
|
// check if it has been cached.
|
|
cachedStructInfo, ok := cf.getCachedConvertStructInfo(structType)
|
|
if ok {
|
|
// directly returns the cached struct info if already exists.
|
|
return cachedStructInfo
|
|
}
|
|
|
|
// else create one.
|
|
|
|
// it parses and generates a cache info for given struct type.
|
|
cachedStructInfo = NewCachedStructInfo(cf)
|
|
var (
|
|
priorityTagArray []string
|
|
parentIndex = make([]int, 0)
|
|
)
|
|
if priorityTag != "" {
|
|
priorityTagArray = append(utils.SplitAndTrim(priorityTag, ","), gtag.StructTagPriority...)
|
|
} else {
|
|
priorityTagArray = gtag.StructTagPriority
|
|
}
|
|
cf.parseStructToCachedStructInfo(structType, parentIndex, cachedStructInfo, priorityTagArray)
|
|
cf.storeCachedStructInfo(structType, cachedStructInfo)
|
|
return cachedStructInfo
|
|
}
|
|
|
|
func (cf *Converter) storeCachedStructInfo(structType reflect.Type, cachedStructInfo *CachedStructInfo) {
|
|
// Temporarily enabled as an experimental feature
|
|
cf.cachedStructsInfoMap.Store(structType, cachedStructInfo)
|
|
}
|
|
|
|
func (cf *Converter) getCachedConvertStructInfo(structType reflect.Type) (*CachedStructInfo, bool) {
|
|
// Temporarily enabled as an experimental feature
|
|
v, ok := cf.cachedStructsInfoMap.Load(structType)
|
|
if ok {
|
|
return v.(*CachedStructInfo), ok
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
// parseStructToCachedStructInfo parses given struct reflection type and stores its fields info into given CachedStructInfo.
|
|
// It stores nothing into CachedStructInfo if given struct reflection type has no fields.
|
|
func (cf *Converter) parseStructToCachedStructInfo(
|
|
structType reflect.Type,
|
|
fieldIndexes []int,
|
|
cachedStructInfo *CachedStructInfo,
|
|
priorityTagArray []string,
|
|
) {
|
|
var (
|
|
fieldName string
|
|
structField reflect.StructField
|
|
fieldType reflect.Type
|
|
)
|
|
// TODO:
|
|
// Check if the structure has already been cached in the cache.
|
|
// If it has been cached, some information can be reused,
|
|
// but the [FieldIndex] needs to be reset.
|
|
// We will not implement it temporarily because it is somewhat complex
|
|
for i := 0; i < structType.NumField(); i++ {
|
|
structField = structType.Field(i)
|
|
fieldType = structField.Type
|
|
fieldName = structField.Name
|
|
// Only do converting to public attributes.
|
|
if !utils.IsLetterUpper(fieldName[0]) {
|
|
continue
|
|
}
|
|
|
|
copyFieldIndexes := make([]int, len(fieldIndexes))
|
|
copy(copyFieldIndexes, fieldIndexes)
|
|
|
|
// normal basic attributes.
|
|
if structField.Anonymous {
|
|
// handle struct attributes, it might be struct/*struct embedded..
|
|
if fieldType.Kind() == reflect.Ptr {
|
|
fieldType = fieldType.Elem()
|
|
}
|
|
if fieldType.Kind() != reflect.Struct {
|
|
continue
|
|
}
|
|
// Skip the embedded structure of the 0 field,
|
|
if fieldType.NumField() == 0 {
|
|
continue
|
|
}
|
|
if structField.Tag != "" {
|
|
// Do not add anonymous structures without tags
|
|
cachedStructInfo.AddField(structField, append(copyFieldIndexes, i), priorityTagArray)
|
|
}
|
|
cf.parseStructToCachedStructInfo(fieldType, append(copyFieldIndexes, i), cachedStructInfo, priorityTagArray)
|
|
continue
|
|
}
|
|
// Do not directly use append(fieldIndexes, i)
|
|
// When the structure is nested deeply, it may lead to bugs,
|
|
// which are caused by the slice expansion mechanism
|
|
// So it is necessary to allocate a separate index for each field
|
|
// See details https://github.com/gogf/gf/issues/3789
|
|
cachedStructInfo.AddField(structField, append(copyFieldIndexes, i), priorityTagArray)
|
|
}
|
|
}
|