1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-11 07:17:13 +00:00
Beego/orm/orm_querym2m.go

145 lines
3.8 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.
package orm
2016-01-14 23:49:28 +08:00
import "reflect"
2014-01-17 23:28:54 +08:00
// model to model struct
type queryM2M struct {
md interface{}
mi *modelInfo
fi *fieldInfo
qs *querySet
ind reflect.Value
}
2014-01-17 23:28:54 +08:00
// add models to origin models when creating queryM2M.
// example:
// m2m := orm.QueryM2M(post,"Tag")
// m2m.Add(&Tag1{},&Tag2{})
// for _,tag := range post.Tags{}
//
// make sure the relation is defined in post model struct tag.
func (o *queryM2M) Add(mds ...interface{}) (int64, error) {
fi := o.fi
mi := fi.relThroughModelInfo
mfi := fi.reverseFieldInfo
rfi := fi.reverseFieldInfoTwo
orm := o.qs.orm
dbase := orm.alias.DbBaser
var models []interface{}
2016-01-18 00:18:21 +08:00
var otherValues []interface{}
var otherNames []string
2015-11-19 16:46:14 +02:00
for _, colname := range mi.fields.dbcols {
2016-01-14 23:49:28 +08:00
if colname != mfi.column && colname != rfi.column && colname != fi.mi.fields.pk.column &&
mi.fields.columns[colname] != mi.fields.pk {
2016-01-18 00:18:21 +08:00
otherNames = append(otherNames, colname)
2015-11-19 16:46:14 +02:00
}
}
for i, md := range mds {
if reflect.Indirect(reflect.ValueOf(md)).Kind() != reflect.Struct && i > 0 {
2016-01-18 00:18:21 +08:00
otherValues = append(otherValues, md)
2015-12-02 16:35:36 +08:00
mds = append(mds[:i], mds[i+1:]...)
2015-11-19 16:46:14 +02:00
}
2015-11-19 14:30:14 +02:00
}
for _, md := range mds {
val := reflect.ValueOf(md)
if val.Kind() == reflect.Slice || val.Kind() == reflect.Array {
for i := 0; i < val.Len(); i++ {
v := val.Index(i)
if v.CanInterface() {
models = append(models, v.Interface())
}
}
} else {
models = append(models, md)
}
}
_, v1, exist := getExistPk(o.mi, o.ind)
if exist == false {
panic(ErrMissPK)
}
2015-11-19 16:46:14 +02:00
names := []string{mfi.column, rfi.column}
2014-01-06 11:07:03 +08:00
2015-11-19 16:46:14 +02:00
values := make([]interface{}, 0, len(models)*2)
for _, md := range models {
ind := reflect.Indirect(reflect.ValueOf(md))
var v2 interface{}
if ind.Kind() != reflect.Struct {
v2 = ind.Interface()
} else {
_, v2, exist = getExistPk(fi.relModelInfo, ind)
if exist == false {
panic(ErrMissPK)
}
}
2015-11-19 16:46:14 +02:00
values = append(values, v1, v2)
}
2016-01-18 00:18:21 +08:00
names = append(names, otherNames...)
values = append(values, otherValues...)
2014-01-06 11:07:03 +08:00
return dbase.InsertValue(orm.db, mi, true, names, values)
}
2014-01-17 23:28:54 +08:00
// remove models following the origin model relationship
func (o *queryM2M) Remove(mds ...interface{}) (int64, error) {
fi := o.fi
qs := o.qs.Filter(fi.reverseFieldInfo.name, o.md)
nums, err := qs.Filter(fi.reverseFieldInfoTwo.name+ExprSep+"in", mds).Delete()
if err != nil {
return nums, err
}
return nums, nil
}
2014-01-17 23:28:54 +08:00
// check model is existed in relationship of origin model
func (o *queryM2M) Exist(md interface{}) bool {
fi := o.fi
return o.qs.Filter(fi.reverseFieldInfo.name, o.md).
Filter(fi.reverseFieldInfoTwo.name, md).Exist()
}
2014-01-17 23:28:54 +08:00
// clean all models in related of origin model
func (o *queryM2M) Clear() (int64, error) {
fi := o.fi
return o.qs.Filter(fi.reverseFieldInfo.name, o.md).Delete()
}
2014-01-17 23:28:54 +08:00
// count all related models of origin model
func (o *queryM2M) Count() (int64, error) {
fi := o.fi
return o.qs.Filter(fi.reverseFieldInfo.name, o.md).Count()
}
var _ QueryM2Mer = new(queryM2M)
2014-01-17 23:28:54 +08:00
// create new M2M queryer.
func newQueryM2M(md interface{}, o *orm, mi *modelInfo, fi *fieldInfo, ind reflect.Value) QueryM2Mer {
qm2m := new(queryM2M)
qm2m.md = md
qm2m.mi = mi
qm2m.fi = fi
qm2m.ind = ind
qm2m.qs = newQuerySet(o, fi.relThroughModelInfo).(*querySet)
return qm2m
}