149 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2014 beego Author. All Rights Reserved.
 | |
| //
 | |
| // 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
 | |
| //
 | |
| //      http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // 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.
 | |
| 
 | |
| package orm
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"reflect"
 | |
| )
 | |
| 
 | |
| // single model info
 | |
| type modelInfo struct {
 | |
| 	manual    bool
 | |
| 	isThrough bool
 | |
| 	pkg       string
 | |
| 	name      string
 | |
| 	fullName  string
 | |
| 	table     string
 | |
| 	model     interface{}
 | |
| 	fields    *fields
 | |
| 	addrField reflect.Value // store the original struct value
 | |
| 	uniques   []string
 | |
| }
 | |
| 
 | |
| // new model info
 | |
| func newModelInfo(val reflect.Value) (mi *modelInfo) {
 | |
| 	mi = &modelInfo{}
 | |
| 	mi.fields = newFields()
 | |
| 	ind := reflect.Indirect(val)
 | |
| 	mi.addrField = val
 | |
| 	mi.name = ind.Type().Name()
 | |
| 	mi.fullName = getFullName(ind.Type())
 | |
| 	addModelFields(mi, ind, "", []int{})
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // index: FieldByIndex returns the nested field corresponding to index
 | |
| func addModelFields(mi *modelInfo, ind reflect.Value, mName string, index []int) {
 | |
| 	var (
 | |
| 		err error
 | |
| 		fi  *fieldInfo
 | |
| 		sf  reflect.StructField
 | |
| 	)
 | |
| 
 | |
| 	for i := 0; i < ind.NumField(); i++ {
 | |
| 		field := ind.Field(i)
 | |
| 		sf = ind.Type().Field(i)
 | |
| 		// if the field is unexported skip
 | |
| 		if sf.PkgPath != "" {
 | |
| 			continue
 | |
| 		}
 | |
| 		// add anonymous struct fields
 | |
| 		if sf.Anonymous {
 | |
| 			addModelFields(mi, field, mName+"."+sf.Name, append(index, i))
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		fi, err = newFieldInfo(mi, field, sf, mName)
 | |
| 		if err == errSkipField {
 | |
| 			err = nil
 | |
| 			continue
 | |
| 		} else if err != nil {
 | |
| 			break
 | |
| 		}
 | |
| 		// record current field index
 | |
| 		fi.fieldIndex = append(fi.fieldIndex, index...)
 | |
| 		fi.fieldIndex = append(fi.fieldIndex, i)
 | |
| 		fi.mi = mi
 | |
| 		fi.inModel = true
 | |
| 		if !mi.fields.Add(fi) {
 | |
| 			err = fmt.Errorf("duplicate column name: %s", fi.column)
 | |
| 			break
 | |
| 		}
 | |
| 		if fi.pk {
 | |
| 			if mi.fields.pk != nil {
 | |
| 				err = fmt.Errorf("one model must have one pk field only")
 | |
| 				break
 | |
| 			} else {
 | |
| 				mi.fields.pk = fi
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if err != nil {
 | |
| 		fmt.Println(fmt.Errorf("field: %s.%s, %s", ind.Type(), sf.Name, err))
 | |
| 		os.Exit(2)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // combine related model info to new model info.
 | |
| // prepare for relation models query.
 | |
| func newM2MModelInfo(m1, m2 *modelInfo) (mi *modelInfo) {
 | |
| 	mi = new(modelInfo)
 | |
| 	mi.fields = newFields()
 | |
| 	mi.table = m1.table + "_" + m2.table + "s"
 | |
| 	mi.name = camelString(mi.table)
 | |
| 	mi.fullName = m1.pkg + "." + mi.name
 | |
| 
 | |
| 	fa := new(fieldInfo) // pk
 | |
| 	f1 := new(fieldInfo) // m1 table RelForeignKey
 | |
| 	f2 := new(fieldInfo) // m2 table RelForeignKey
 | |
| 	fa.fieldType = TypeBigIntegerField
 | |
| 	fa.auto = true
 | |
| 	fa.pk = true
 | |
| 	fa.dbcol = true
 | |
| 	fa.name = "Id"
 | |
| 	fa.column = "id"
 | |
| 	fa.fullName = mi.fullName + "." + fa.name
 | |
| 
 | |
| 	f1.dbcol = true
 | |
| 	f2.dbcol = true
 | |
| 	f1.fieldType = RelForeignKey
 | |
| 	f2.fieldType = RelForeignKey
 | |
| 	f1.name = camelString(m1.table)
 | |
| 	f2.name = camelString(m2.table)
 | |
| 	f1.fullName = mi.fullName + "." + f1.name
 | |
| 	f2.fullName = mi.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 = mi
 | |
| 	f2.mi = mi
 | |
| 
 | |
| 	mi.fields.Add(fa)
 | |
| 	mi.fields.Add(f1)
 | |
| 	mi.fields.Add(f2)
 | |
| 	mi.fields.pk = fa
 | |
| 
 | |
| 	mi.uniques = []string{f1.column, f2.column}
 | |
| 	return
 | |
| }
 |