goext/wmo/reflection.go

151 lines
3.2 KiB
Go
Raw Normal View History

2023-06-06 21:18:40 +02:00
package wmo
import (
"gogs.mikescher.com/BlackForestBytes/goext/langext"
2023-06-07 16:58:17 +02:00
"gogs.mikescher.com/BlackForestBytes/goext/reflectext"
2023-06-06 21:18:40 +02:00
"reflect"
2023-06-07 17:48:36 +02:00
"strings"
2023-06-06 21:18:40 +02:00
)
2023-06-10 18:35:56 +02:00
func (c *Coll[TData]) EnsureInitializedReflection(v TData) {
if !c.isInterfaceDataType {
return // only dynamically load dataTypeMap on interface TData
}
rval := reflect.ValueOf(v)
for rval.Type().Kind() == reflect.Pointer {
rval = rval.Elem()
}
if _, ok := c.implDataTypeMap[rval.Type()]; ok {
return // already loaded
}
m := make(map[string]fullTypeRef)
c.initFields("", rval, m, make([]int, 0))
2023-06-06 21:18:40 +02:00
2023-06-10 18:35:56 +02:00
c.implDataTypeMap[rval.Type()] = m
}
func (c *Coll[TData]) init() {
2023-06-06 21:18:40 +02:00
example := *new(TData)
2023-06-10 18:35:56 +02:00
datatype := reflect.TypeOf(&example).Elem()
if datatype.Kind() == reflect.Interface {
c.isInterfaceDataType = true
2023-06-06 21:18:40 +02:00
2023-06-10 18:35:56 +02:00
c.dataTypeMap = make(map[string]fullTypeRef)
c.implDataTypeMap = make(map[reflect.Type]map[string]fullTypeRef)
} else {
c.isInterfaceDataType = false
c.dataTypeMap = make(map[string]fullTypeRef)
c.implDataTypeMap = make(map[reflect.Type]map[string]fullTypeRef)
v := reflect.ValueOf(example)
c.initFields("", v, c.dataTypeMap, make([]int, 0))
}
2023-06-06 21:18:40 +02:00
}
2023-06-10 18:35:56 +02:00
func (c *Coll[TData]) initFields(prefix string, rval reflect.Value, m map[string]fullTypeRef, idxarr []int) {
2023-06-06 21:18:40 +02:00
rtyp := rval.Type()
for i := 0; i < rtyp.NumField(); i++ {
rsfield := rtyp.Field(i)
rvfield := rval.Field(i)
if !rsfield.IsExported() {
continue
}
bsonkey, found := rsfield.Tag.Lookup("bson")
2023-06-07 17:48:36 +02:00
if !found {
continue
}
if strings.Contains(bsonkey, ",") {
bsonkey = bsonkey[:strings.Index(bsonkey, ",")]
}
if bsonkey == "-" {
2023-06-06 21:18:40 +02:00
continue
}
fullKey := prefix + bsonkey
newIdxArr := langext.ArrCopy(idxarr)
newIdxArr = append(newIdxArr, i)
if rvfield.Type().Kind() == reflect.Pointer {
2023-06-10 18:35:56 +02:00
m[fullKey] = fullTypeRef{
2023-06-06 21:18:40 +02:00
IsPointer: true,
2023-06-07 16:58:17 +02:00
RealType: rvfield.Type(),
2023-06-06 21:18:40 +02:00
Kind: rvfield.Type().Elem().Kind(),
Type: rvfield.Type().Elem(),
2023-06-07 16:58:17 +02:00
UnderlyingType: reflectext.Underlying(rvfield.Type().Elem()),
2023-06-06 21:18:40 +02:00
Name: rsfield.Name,
Index: newIdxArr,
}
} else {
2023-06-10 18:35:56 +02:00
m[fullKey] = fullTypeRef{
2023-06-06 21:18:40 +02:00
IsPointer: false,
2023-06-07 16:58:17 +02:00
RealType: rvfield.Type(),
2023-06-06 21:18:40 +02:00
Kind: rvfield.Type().Kind(),
Type: rvfield.Type(),
2023-06-07 16:58:17 +02:00
UnderlyingType: reflectext.Underlying(rvfield.Type()),
2023-06-06 21:18:40 +02:00
Name: rsfield.Name,
Index: newIdxArr,
}
}
if rvfield.Kind() == reflect.Struct {
2023-06-10 18:35:56 +02:00
c.initFields(fullKey+".", rvfield, m, newIdxArr)
2023-06-06 21:18:40 +02:00
}
}
}
func (c *Coll[TData]) getTokenValueAsMongoType(value string, fieldName string) (any, error) {
2023-06-07 16:58:17 +02:00
fref := c.dataTypeMap[fieldName]
2023-06-06 21:18:40 +02:00
2023-06-07 16:58:17 +02:00
pss := reflectext.PrimitiveStringSerializer{}
2023-06-06 21:18:40 +02:00
2023-06-07 16:58:17 +02:00
return pss.ValueFromString(value, fref.RealType)
2023-06-06 21:18:40 +02:00
}
func (c *Coll[TData]) getFieldValueAsTokenString(entity TData, fieldName string) (string, error) {
realValue := c.getFieldValue(entity, fieldName)
2023-06-07 16:58:17 +02:00
pss := reflectext.PrimitiveStringSerializer{}
2023-06-06 21:18:40 +02:00
2023-06-07 16:58:17 +02:00
return pss.ValueToString(realValue)
2023-06-06 21:18:40 +02:00
}
2023-06-10 18:35:56 +02:00
func (c *Coll[TData]) getFieldType(fieldName string) fullTypeRef {
2023-06-06 21:18:40 +02:00
return c.dataTypeMap[fieldName]
}
func (c *Coll[TData]) getFieldValue(data TData, fieldName string) any {
fref := c.dataTypeMap[fieldName]
rval := reflect.ValueOf(data)
return rval.FieldByIndex(fref.Index).Interface()
}