gorm/utils.go

139 lines
2.8 KiB
Go
Raw Normal View History

2013-10-25 14:04:48 +04:00
package gorm
2013-10-26 05:49:40 +04:00
import (
"bytes"
2014-07-25 12:51:54 +04:00
"go/ast"
2013-11-14 14:59:11 +04:00
"reflect"
2013-10-26 05:49:40 +04:00
"strings"
2013-12-12 16:57:10 +04:00
"sync"
2013-10-26 05:49:40 +04:00
)
2013-12-15 06:09:19 +04:00
type safeMap struct {
m map[string]string
l *sync.RWMutex
2013-12-12 16:57:10 +04:00
}
2013-12-15 06:09:19 +04:00
func (s *safeMap) Set(key string, value string) {
s.l.Lock()
defer s.l.Unlock()
s.m[key] = value
}
func (s *safeMap) Get(key string) string {
s.l.RLock()
defer s.l.RUnlock()
return s.m[key]
}
2014-07-15 07:48:30 +04:00
func FieldByName(name string, value interface{}, withAddr ...bool) (interface{}, bool) {
data := reflect.Indirect(reflect.ValueOf(value))
name = SnakeToUpperCamel(name)
2014-07-15 07:48:30 +04:00
if data.Kind() == reflect.Struct {
if field := data.FieldByName(name); field.IsValid() {
if len(withAddr) > 0 && field.CanAddr() {
return field.Addr().Interface(), true
} else {
return field.Interface(), true
}
}
} else if data.Kind() == reflect.Slice {
elem := data.Type().Elem()
if elem.Kind() == reflect.Ptr {
return nil, reflect.New(data.Type().Elem().Elem()).Elem().FieldByName(name).IsValid()
} else {
return nil, reflect.New(data.Type().Elem()).Elem().FieldByName(name).IsValid()
}
}
return nil, false
}
2013-12-15 06:09:19 +04:00
func newSafeMap() *safeMap {
return &safeMap{l: new(sync.RWMutex), m: make(map[string]string)}
}
var smap = newSafeMap()
var umap = newSafeMap()
2013-11-17 04:28:30 +04:00
func ToSnake(u string) string {
2013-12-15 06:09:19 +04:00
if v := smap.Get(u); v != "" {
2013-11-17 04:28:30 +04:00
return v
}
buf := bytes.NewBufferString("")
2013-11-17 04:28:30 +04:00
for i, v := range u {
if i > 0 && v >= 'A' && v <= 'Z' {
buf.WriteRune('_')
}
buf.WriteRune(v)
}
2013-11-17 04:28:30 +04:00
s := strings.ToLower(buf.String())
2013-12-15 06:09:19 +04:00
go smap.Set(u, s)
2013-11-17 04:28:30 +04:00
return s
}
func SnakeToUpperCamel(s string) string {
2013-12-15 06:09:19 +04:00
if v := umap.Get(s); v != "" {
2013-11-17 04:28:30 +04:00
return v
}
buf := bytes.NewBufferString("")
for _, v := range strings.Split(s, "_") {
if len(v) > 0 {
buf.WriteString(strings.ToUpper(v[:1]))
buf.WriteString(v[1:])
}
}
2013-11-17 04:28:30 +04:00
u := buf.String()
2013-12-15 06:09:19 +04:00
go umap.Set(s, u)
2013-11-17 04:28:30 +04:00
return u
}
2014-07-25 12:51:54 +04:00
func GetPrimaryKey(value interface{}) string {
var indirectValue = reflect.Indirect(reflect.ValueOf(value))
2014-01-26 08:41:37 +04:00
2014-07-25 12:51:54 +04:00
if indirectValue.Kind() == reflect.Slice {
indirectValue = reflect.New(indirectValue.Type().Elem()).Elem()
2013-11-14 14:59:11 +04:00
}
2014-01-27 18:36:08 +04:00
2014-07-25 12:51:54 +04:00
if indirectValue.IsValid() {
hasId := false
2014-07-25 12:51:54 +04:00
scopeTyp := indirectValue.Type()
for i := 0; i < scopeTyp.NumField(); i++ {
fieldStruct := scopeTyp.Field(i)
if !ast.IsExported(fieldStruct.Name) {
continue
2014-01-27 18:36:08 +04:00
}
2014-07-29 14:21:36 +04:00
settings := parseTagSetting(fieldStruct.Tag.Get("gorm"))
if _, ok := settings["PRIMARY_KEY"]; ok {
2014-07-25 12:51:54 +04:00
return fieldStruct.Name
} else if fieldStruct.Name == "Id" {
hasId = true
2014-01-27 18:36:08 +04:00
}
}
if hasId {
return "Id"
}
2014-01-27 18:36:08 +04:00
}
2014-07-25 12:51:54 +04:00
return ""
2014-01-27 18:36:08 +04:00
}
func parseTagSetting(str string) map[string]string {
tags := strings.Split(str, ";")
setting := map[string]string{}
for _, value := range tags {
v := strings.Split(value, ":")
k := strings.TrimSpace(strings.ToUpper(v[0]))
if len(v) == 2 {
setting[k] = v[1]
} else {
setting[k] = k
}
}
return setting
}