227 lines
4.3 KiB
Go
227 lines
4.3 KiB
Go
|
package storm
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"reflect"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/asdine/storm/v3/index"
|
||
|
bolt "go.etcd.io/bbolt"
|
||
|
)
|
||
|
|
||
|
// Storm tags
|
||
|
const (
|
||
|
tagID = "id"
|
||
|
tagIdx = "index"
|
||
|
tagUniqueIdx = "unique"
|
||
|
tagInline = "inline"
|
||
|
tagIncrement = "increment"
|
||
|
indexPrefix = "__storm_index_"
|
||
|
)
|
||
|
|
||
|
type fieldConfig struct {
|
||
|
Name string
|
||
|
Index string
|
||
|
IsZero bool
|
||
|
IsID bool
|
||
|
Increment bool
|
||
|
IncrementStart int64
|
||
|
IsInteger bool
|
||
|
Value *reflect.Value
|
||
|
ForceUpdate bool
|
||
|
}
|
||
|
|
||
|
// structConfig is a structure gathering all the relevant informations about a model
|
||
|
type structConfig struct {
|
||
|
Name string
|
||
|
Fields map[string]*fieldConfig
|
||
|
ID *fieldConfig
|
||
|
}
|
||
|
|
||
|
func extract(s *reflect.Value, mi ...*structConfig) (*structConfig, error) {
|
||
|
if s.Kind() == reflect.Ptr {
|
||
|
e := s.Elem()
|
||
|
s = &e
|
||
|
}
|
||
|
if s.Kind() != reflect.Struct {
|
||
|
return nil, ErrBadType
|
||
|
}
|
||
|
|
||
|
typ := s.Type()
|
||
|
|
||
|
var child bool
|
||
|
|
||
|
var m *structConfig
|
||
|
if len(mi) > 0 {
|
||
|
m = mi[0]
|
||
|
child = true
|
||
|
} else {
|
||
|
m = &structConfig{}
|
||
|
m.Fields = make(map[string]*fieldConfig)
|
||
|
}
|
||
|
|
||
|
if m.Name == "" {
|
||
|
m.Name = typ.Name()
|
||
|
}
|
||
|
|
||
|
numFields := s.NumField()
|
||
|
for i := 0; i < numFields; i++ {
|
||
|
field := typ.Field(i)
|
||
|
value := s.Field(i)
|
||
|
|
||
|
if field.PkgPath != "" {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
err := extractField(&value, &field, m, child)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if child {
|
||
|
return m, nil
|
||
|
}
|
||
|
|
||
|
if m.ID == nil {
|
||
|
return nil, ErrNoID
|
||
|
}
|
||
|
|
||
|
if m.Name == "" {
|
||
|
return nil, ErrNoName
|
||
|
}
|
||
|
|
||
|
return m, nil
|
||
|
}
|
||
|
|
||
|
func extractField(value *reflect.Value, field *reflect.StructField, m *structConfig, isChild bool) error {
|
||
|
var f *fieldConfig
|
||
|
var err error
|
||
|
|
||
|
tag := field.Tag.Get("storm")
|
||
|
if tag != "" {
|
||
|
f = &fieldConfig{
|
||
|
Name: field.Name,
|
||
|
IsZero: isZero(value),
|
||
|
IsInteger: isInteger(value),
|
||
|
Value: value,
|
||
|
IncrementStart: 1,
|
||
|
}
|
||
|
|
||
|
tags := strings.Split(tag, ",")
|
||
|
|
||
|
for _, tag := range tags {
|
||
|
switch tag {
|
||
|
case "id":
|
||
|
f.IsID = true
|
||
|
f.Index = tagUniqueIdx
|
||
|
case tagUniqueIdx, tagIdx:
|
||
|
f.Index = tag
|
||
|
case tagInline:
|
||
|
if value.Kind() == reflect.Ptr {
|
||
|
e := value.Elem()
|
||
|
value = &e
|
||
|
}
|
||
|
if value.Kind() == reflect.Struct {
|
||
|
a := value.Addr()
|
||
|
_, err := extract(&a, m)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
// we don't need to save this field
|
||
|
return nil
|
||
|
default:
|
||
|
if strings.HasPrefix(tag, tagIncrement) {
|
||
|
f.Increment = true
|
||
|
parts := strings.Split(tag, "=")
|
||
|
if parts[0] != tagIncrement {
|
||
|
return ErrUnknownTag
|
||
|
}
|
||
|
if len(parts) > 1 {
|
||
|
f.IncrementStart, err = strconv.ParseInt(parts[1], 0, 64)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
return ErrUnknownTag
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if _, ok := m.Fields[f.Name]; !ok || !isChild {
|
||
|
m.Fields[f.Name] = f
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if m.ID == nil && f != nil && f.IsID {
|
||
|
m.ID = f
|
||
|
}
|
||
|
|
||
|
// the field is named ID and no ID field has been detected before
|
||
|
if m.ID == nil && field.Name == "ID" {
|
||
|
if f == nil {
|
||
|
f = &fieldConfig{
|
||
|
Index: tagUniqueIdx,
|
||
|
Name: field.Name,
|
||
|
IsZero: isZero(value),
|
||
|
IsInteger: isInteger(value),
|
||
|
IsID: true,
|
||
|
Value: value,
|
||
|
IncrementStart: 1,
|
||
|
}
|
||
|
m.Fields[field.Name] = f
|
||
|
}
|
||
|
m.ID = f
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func extractSingleField(ref *reflect.Value, fieldName string) (*structConfig, error) {
|
||
|
var cfg structConfig
|
||
|
cfg.Fields = make(map[string]*fieldConfig)
|
||
|
|
||
|
f, ok := ref.Type().FieldByName(fieldName)
|
||
|
if !ok || f.PkgPath != "" {
|
||
|
return nil, fmt.Errorf("field %s not found", fieldName)
|
||
|
}
|
||
|
|
||
|
v := ref.FieldByName(fieldName)
|
||
|
err := extractField(&v, &f, &cfg, false)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &cfg, nil
|
||
|
}
|
||
|
|
||
|
func getIndex(bucket *bolt.Bucket, idxKind string, fieldName string) (index.Index, error) {
|
||
|
var idx index.Index
|
||
|
var err error
|
||
|
|
||
|
switch idxKind {
|
||
|
case tagUniqueIdx:
|
||
|
idx, err = index.NewUniqueIndex(bucket, []byte(indexPrefix+fieldName))
|
||
|
case tagIdx:
|
||
|
idx, err = index.NewListIndex(bucket, []byte(indexPrefix+fieldName))
|
||
|
default:
|
||
|
err = ErrIdxNotFound
|
||
|
}
|
||
|
|
||
|
return idx, err
|
||
|
}
|
||
|
|
||
|
func isZero(v *reflect.Value) bool {
|
||
|
zero := reflect.Zero(v.Type()).Interface()
|
||
|
current := v.Interface()
|
||
|
return reflect.DeepEqual(current, zero)
|
||
|
}
|
||
|
|
||
|
func isInteger(v *reflect.Value) bool {
|
||
|
kind := v.Kind()
|
||
|
return v != nil && kind >= reflect.Int && kind <= reflect.Uint64
|
||
|
}
|