1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-15 01:57:12 +00:00
Beego/orm/models_utils.go

208 lines
5.1 KiB
Go
Raw Normal View History

2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2013-07-30 20:32:38 +08:00
package orm
import (
"database/sql"
2013-07-30 20:32:38 +08:00
"fmt"
"reflect"
"strings"
"time"
)
2014-01-17 23:28:54 +08:00
// get reflect.Type name with package path.
func getFullName(typ reflect.Type) string {
return typ.PkgPath() + "." + typ.Name()
}
2014-01-17 23:28:54 +08:00
// get table name. method, or field name. auto snaked.
func getTableName(val reflect.Value) string {
2013-07-30 20:32:38 +08:00
ind := reflect.Indirect(val)
fun := val.MethodByName("TableName")
if fun.IsValid() {
vals := fun.Call([]reflect.Value{})
if len(vals) > 0 {
val := vals[0]
if val.Kind() == reflect.String {
return val.String()
}
}
}
return snakeString(ind.Type().Name())
}
2014-01-17 23:28:54 +08:00
// get table engine, mysiam or innodb.
func getTableEngine(val reflect.Value) string {
fun := val.MethodByName("TableEngine")
if fun.IsValid() {
vals := fun.Call([]reflect.Value{})
if len(vals) > 0 {
val := vals[0]
if val.Kind() == reflect.String {
return val.String()
}
}
}
return ""
}
2014-01-17 23:28:54 +08:00
// get table index from method.
func getTableIndex(val reflect.Value) [][]string {
fun := val.MethodByName("TableIndex")
if fun.IsValid() {
vals := fun.Call([]reflect.Value{})
if len(vals) > 0 {
val := vals[0]
if val.CanInterface() {
if d, ok := val.Interface().([][]string); ok {
return d
}
}
}
}
return nil
}
2014-01-17 23:28:54 +08:00
// get table unique from method
func getTableUnique(val reflect.Value) [][]string {
fun := val.MethodByName("TableUnique")
if fun.IsValid() {
vals := fun.Call([]reflect.Value{})
if len(vals) > 0 {
val := vals[0]
if val.CanInterface() {
if d, ok := val.Interface().([][]string); ok {
return d
}
}
}
}
return nil
}
2014-01-17 23:28:54 +08:00
// get snaked column name
2013-07-30 20:32:38 +08:00
func getColumnName(ft int, addrField reflect.Value, sf reflect.StructField, col string) string {
2013-11-06 22:05:10 +08:00
column := col
if col == "" {
2013-07-30 20:32:38 +08:00
column = snakeString(sf.Name)
}
switch ft {
case RelForeignKey, RelOneToOne:
2013-11-06 22:05:10 +08:00
if len(col) == 0 {
column = column + "_id"
}
2013-07-30 20:32:38 +08:00
case RelManyToMany, RelReverseMany, RelReverseOne:
column = sf.Name
}
return column
}
2014-01-17 23:28:54 +08:00
// return field type as type constant from reflect.Value
2013-07-30 20:32:38 +08:00
func getFieldType(val reflect.Value) (ft int, err error) {
switch val.Type() {
case reflect.TypeOf(new(int8)):
ft = TypeBitField
case reflect.TypeOf(new(int16)):
2013-07-30 20:32:38 +08:00
ft = TypeSmallIntegerField
case reflect.TypeOf(new(int32)),
reflect.TypeOf(new(int)):
2013-07-30 20:32:38 +08:00
ft = TypeIntegerField
case reflect.TypeOf(new(int64)):
2013-07-30 20:32:38 +08:00
ft = TypeBigIntegerField
case reflect.TypeOf(new(uint8)):
2013-08-19 22:37:39 +08:00
ft = TypePositiveBitField
case reflect.TypeOf(new(uint16)):
2013-07-30 20:32:38 +08:00
ft = TypePositiveSmallIntegerField
case reflect.TypeOf(new(uint32)),
reflect.TypeOf(new(uint)):
2013-07-30 20:32:38 +08:00
ft = TypePositiveIntegerField
case reflect.TypeOf(new(uint64)):
2013-07-30 20:32:38 +08:00
ft = TypePositiveBigIntegerField
case reflect.TypeOf(new(float32)),
reflect.TypeOf(new(float64)):
2013-07-30 20:32:38 +08:00
ft = TypeFloatField
case reflect.TypeOf(new(bool)):
2013-07-30 20:32:38 +08:00
ft = TypeBooleanField
case reflect.TypeOf(new(string)):
ft = TypeCharField
2013-07-30 20:32:38 +08:00
default:
elm := reflect.Indirect(val)
switch elm.Kind() {
case reflect.Int8:
ft = TypeBitField
case reflect.Int16:
ft = TypeSmallIntegerField
case reflect.Int32, reflect.Int:
ft = TypeIntegerField
case reflect.Int64:
ft = TypeBigIntegerField
case reflect.Uint8:
ft = TypePositiveBitField
case reflect.Uint16:
ft = TypePositiveSmallIntegerField
case reflect.Uint32, reflect.Uint:
ft = TypePositiveIntegerField
case reflect.Uint64:
ft = TypePositiveBigIntegerField
case reflect.Float32, reflect.Float64:
ft = TypeFloatField
case reflect.Bool:
ft = TypeBooleanField
case reflect.String:
ft = TypeCharField
default:
if elm.Interface() == nil {
panic(fmt.Errorf("%s is nil pointer, may be miss setting tag", val))
}
switch elm.Interface().(type) {
case sql.NullInt64:
ft = TypeBigIntegerField
case sql.NullFloat64:
ft = TypeFloatField
case sql.NullBool:
ft = TypeBooleanField
case sql.NullString:
ft = TypeCharField
case time.Time:
ft = TypeDateTimeField
}
2013-07-30 20:32:38 +08:00
}
}
if ft&IsFieldType == 0 {
err = fmt.Errorf("unsupport field type %s, may be miss setting tag", val)
}
return
}
2014-01-17 23:28:54 +08:00
// parse struct tag string
2013-07-30 20:32:38 +08:00
func parseStructTag(data string, attrs *map[string]bool, tags *map[string]string) {
attr := make(map[string]bool)
tag := make(map[string]string)
for _, v := range strings.Split(data, defaultStructTagDelim) {
2013-07-30 20:32:38 +08:00
v = strings.TrimSpace(v)
if supportTag[v] == 1 {
attr[v] = true
} else if i := strings.Index(v, "("); i > 0 && strings.Index(v, ")") == len(v)-1 {
name := v[:i]
if supportTag[name] == 2 {
v = v[i+1 : len(v)-1]
tag[name] = v
}
}
}
*attrs = attr
*tags = tag
}