1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-12 09:47:11 +00:00
Beego/orm/models_info_m.go

140 lines
2.8 KiB
Go
Raw Normal View History

2014-04-12 13:18:18 +08:00
// Beego (http://beego.me/)
// @description beego is an open-source, high-performance web framework for the Go programming language.
// @link http://github.com/astaxie/beego for the canonical source repository
// @license http://github.com/astaxie/beego/blob/master/LICENSE
// @authors slene
2013-07-30 20:32:38 +08:00
package orm
import (
"errors"
"fmt"
"os"
"reflect"
)
2014-01-17 23:28:54 +08:00
// single model info
2013-07-30 20:32:38 +08:00
type modelInfo struct {
pkg string
name string
fullName string
table string
model interface{}
2013-07-30 20:32:38 +08:00
fields *fields
manual bool
addrField reflect.Value
uniques []string
isThrough bool
2013-07-30 20:32:38 +08:00
}
2014-01-17 23:28:54 +08:00
// new model info
func newModelInfo(val reflect.Value) (info *modelInfo) {
2013-07-30 20:32:38 +08:00
var (
err error
fi *fieldInfo
sf reflect.StructField
)
info = &modelInfo{}
info.fields = newFields()
ind := reflect.Indirect(val)
typ := ind.Type()
info.addrField = val
2013-07-30 20:32:38 +08:00
info.name = typ.Name()
info.fullName = getFullName(typ)
2013-07-30 20:32:38 +08:00
for i := 0; i < ind.NumField(); i++ {
field := ind.Field(i)
sf = ind.Type().Field(i)
if sf.PkgPath != "" {
continue
}
2013-07-30 20:32:38 +08:00
fi, err = newFieldInfo(info, field, sf)
2013-07-30 20:32:38 +08:00
if err != nil {
if err == errSkipField {
err = nil
continue
}
2013-07-30 20:32:38 +08:00
break
}
2013-08-07 19:11:44 +08:00
2013-07-30 20:32:38 +08:00
added := info.fields.Add(fi)
if added == false {
err = errors.New(fmt.Sprintf("duplicate column name: %s", fi.column))
break
}
2013-08-07 19:11:44 +08:00
2013-07-30 20:32:38 +08:00
if fi.pk {
if info.fields.pk != nil {
err = errors.New(fmt.Sprintf("one model must have one pk field only"))
break
} else {
2013-08-07 19:11:44 +08:00
info.fields.pk = fi
2013-07-30 20:32:38 +08:00
}
}
2013-08-07 19:11:44 +08:00
2013-07-30 20:32:38 +08:00
fi.fieldIndex = i
fi.mi = info
2013-09-22 19:20:40 +08:00
fi.inModel = true
2013-07-30 20:32:38 +08:00
}
if err != nil {
fmt.Println(fmt.Errorf("field: %s.%s, %s", ind.Type(), sf.Name, err))
os.Exit(2)
}
return
}
2014-01-17 23:28:54 +08:00
// combine related model info to new model info.
// prepare for relation models query.
2013-07-30 20:32:38 +08:00
func newM2MModelInfo(m1, m2 *modelInfo) (info *modelInfo) {
info = new(modelInfo)
info.fields = newFields()
info.table = m1.table + "_" + m2.table + "s"
2013-07-30 20:32:38 +08:00
info.name = camelString(info.table)
info.fullName = m1.pkg + "." + info.name
fa := new(fieldInfo)
f1 := new(fieldInfo)
f2 := new(fieldInfo)
fa.fieldType = TypeBigIntegerField
fa.auto = true
fa.pk = true
fa.dbcol = true
2013-08-19 22:37:39 +08:00
fa.name = "Id"
fa.column = "id"
fa.fullName = info.fullName + "." + fa.name
2013-07-30 20:32:38 +08:00
f1.dbcol = true
f2.dbcol = true
f1.fieldType = RelForeignKey
f2.fieldType = RelForeignKey
f1.name = camelString(m1.table)
f2.name = camelString(m2.table)
f1.fullName = info.fullName + "." + f1.name
f2.fullName = info.fullName + "." + f2.name
f1.column = m1.table + "_id"
f2.column = m2.table + "_id"
f1.rel = true
f2.rel = true
f1.relTable = m1.table
f2.relTable = m2.table
f1.relModelInfo = m1
f2.relModelInfo = m2
f1.mi = info
f2.mi = info
info.fields.Add(fa)
info.fields.Add(f1)
info.fields.Add(f2)
2013-08-07 19:11:44 +08:00
info.fields.pk = fa
info.uniques = []string{f1.column, f2.column}
2013-07-30 20:32:38 +08:00
return
}