mirror of
https://github.com/astaxie/beego.git
synced 2024-12-23 18:50:49 +00:00
385 lines
7.4 KiB
Go
385 lines
7.4 KiB
Go
|
package orm
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
type dbTable struct {
|
||
|
id int
|
||
|
index string
|
||
|
name string
|
||
|
names []string
|
||
|
sel bool
|
||
|
inner bool
|
||
|
mi *modelInfo
|
||
|
fi *fieldInfo
|
||
|
jtl *dbTable
|
||
|
}
|
||
|
|
||
|
type dbTables struct {
|
||
|
tablesM map[string]*dbTable
|
||
|
tables []*dbTable
|
||
|
mi *modelInfo
|
||
|
base dbBaser
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) set(names []string, mi *modelInfo, fi *fieldInfo, inner bool) *dbTable {
|
||
|
name := strings.Join(names, ExprSep)
|
||
|
if j, ok := t.tablesM[name]; ok {
|
||
|
j.name = name
|
||
|
j.mi = mi
|
||
|
j.fi = fi
|
||
|
j.inner = inner
|
||
|
} else {
|
||
|
i := len(t.tables) + 1
|
||
|
jt := &dbTable{i, fmt.Sprintf("T%d", i), name, names, false, inner, mi, fi, nil}
|
||
|
t.tablesM[name] = jt
|
||
|
t.tables = append(t.tables, jt)
|
||
|
}
|
||
|
return t.tablesM[name]
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) add(names []string, mi *modelInfo, fi *fieldInfo, inner bool) (*dbTable, bool) {
|
||
|
name := strings.Join(names, ExprSep)
|
||
|
if _, ok := t.tablesM[name]; ok == false {
|
||
|
i := len(t.tables) + 1
|
||
|
jt := &dbTable{i, fmt.Sprintf("T%d", i), name, names, false, inner, mi, fi, nil}
|
||
|
t.tablesM[name] = jt
|
||
|
t.tables = append(t.tables, jt)
|
||
|
return jt, true
|
||
|
}
|
||
|
return t.tablesM[name], false
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) get(name string) (*dbTable, bool) {
|
||
|
j, ok := t.tablesM[name]
|
||
|
return j, ok
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) loopDepth(depth int, prefix string, fi *fieldInfo, related []string) []string {
|
||
|
if depth < 0 || fi.fieldType == RelManyToMany {
|
||
|
return related
|
||
|
}
|
||
|
|
||
|
if prefix == "" {
|
||
|
prefix = fi.name
|
||
|
} else {
|
||
|
prefix = prefix + ExprSep + fi.name
|
||
|
}
|
||
|
related = append(related, prefix)
|
||
|
|
||
|
depth--
|
||
|
for _, fi := range fi.relModelInfo.fields.fieldsRel {
|
||
|
related = t.loopDepth(depth, prefix, fi, related)
|
||
|
}
|
||
|
|
||
|
return related
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) parseRelated(rels []string, depth int) {
|
||
|
|
||
|
relsNum := len(rels)
|
||
|
related := make([]string, relsNum)
|
||
|
copy(related, rels)
|
||
|
|
||
|
relDepth := depth
|
||
|
|
||
|
if relsNum != 0 {
|
||
|
relDepth = 0
|
||
|
}
|
||
|
|
||
|
relDepth--
|
||
|
for _, fi := range t.mi.fields.fieldsRel {
|
||
|
related = t.loopDepth(relDepth, "", fi, related)
|
||
|
}
|
||
|
|
||
|
for i, s := range related {
|
||
|
var (
|
||
|
exs = strings.Split(s, ExprSep)
|
||
|
names = make([]string, 0, len(exs))
|
||
|
mmi = t.mi
|
||
|
cansel = true
|
||
|
jtl *dbTable
|
||
|
)
|
||
|
for _, ex := range exs {
|
||
|
if fi, ok := mmi.fields.GetByAny(ex); ok && fi.rel && fi.fieldType != RelManyToMany {
|
||
|
names = append(names, fi.name)
|
||
|
mmi = fi.relModelInfo
|
||
|
|
||
|
jt := t.set(names, mmi, fi, fi.null == false)
|
||
|
jt.jtl = jtl
|
||
|
|
||
|
if fi.reverse {
|
||
|
cansel = false
|
||
|
}
|
||
|
|
||
|
if cansel {
|
||
|
jt.sel = depth > 0
|
||
|
|
||
|
if i < relsNum {
|
||
|
jt.sel = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
jtl = jt
|
||
|
|
||
|
} else {
|
||
|
panic(fmt.Sprintf("unknown model/table name `%s`", ex))
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (t *dbTables) getJoinSql() (join string) {
|
||
|
Q := t.base.TableQuote()
|
||
|
|
||
|
for _, jt := range t.tables {
|
||
|
if jt.inner {
|
||
|
join += "INNER JOIN "
|
||
|
} else {
|
||
|
join += "LEFT OUTER JOIN "
|
||
|
}
|
||
|
var (
|
||
|
table string
|
||
|
t1, t2 string
|
||
|
c1, c2 string
|
||
|
)
|
||
|
t1 = "T0"
|
||
|
if jt.jtl != nil {
|
||
|
t1 = jt.jtl.index
|
||
|
}
|
||
|
t2 = jt.index
|
||
|
table = jt.mi.table
|
||
|
|
||
|
switch {
|
||
|
case jt.fi.fieldType == RelManyToMany || jt.fi.reverse && jt.fi.reverseFieldInfo.fieldType == RelManyToMany:
|
||
|
c1 = jt.fi.mi.fields.pk.column
|
||
|
for _, ffi := range jt.mi.fields.fieldsRel {
|
||
|
if jt.fi.mi == ffi.relModelInfo {
|
||
|
c2 = ffi.column
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
default:
|
||
|
c1 = jt.fi.column
|
||
|
c2 = jt.fi.relModelInfo.fields.pk.column
|
||
|
|
||
|
if jt.fi.reverse {
|
||
|
c1 = jt.mi.fields.pk.column
|
||
|
c2 = jt.fi.reverseFieldInfo.column
|
||
|
}
|
||
|
}
|
||
|
|
||
|
join += fmt.Sprintf("%s%s%s %s ON %s.%s%s%s = %s.%s%s%s ", Q, table, Q, t2,
|
||
|
t2, Q, c2, Q, t1, Q, c1, Q)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (d *dbTables) parseExprs(mi *modelInfo, exprs []string) (index, column, name string, info *fieldInfo, success bool) {
|
||
|
var (
|
||
|
ffi *fieldInfo
|
||
|
jtl *dbTable
|
||
|
mmi = mi
|
||
|
)
|
||
|
|
||
|
num := len(exprs) - 1
|
||
|
names := make([]string, 0)
|
||
|
|
||
|
for i, ex := range exprs {
|
||
|
exist := false
|
||
|
|
||
|
check:
|
||
|
fi, ok := mmi.fields.GetByAny(ex)
|
||
|
|
||
|
if ok {
|
||
|
|
||
|
if num != i {
|
||
|
names = append(names, fi.name)
|
||
|
|
||
|
switch {
|
||
|
case fi.rel:
|
||
|
mmi = fi.relModelInfo
|
||
|
if fi.fieldType == RelManyToMany {
|
||
|
mmi = fi.relThroughModelInfo
|
||
|
}
|
||
|
case fi.reverse:
|
||
|
mmi = fi.reverseFieldInfo.mi
|
||
|
if fi.reverseFieldInfo.fieldType == RelManyToMany {
|
||
|
mmi = fi.reverseFieldInfo.relThroughModelInfo
|
||
|
}
|
||
|
default:
|
||
|
return
|
||
|
}
|
||
|
|
||
|
jt, _ := d.add(names, mmi, fi, fi.null == false)
|
||
|
jt.jtl = jtl
|
||
|
jtl = jt
|
||
|
|
||
|
if fi.rel && fi.fieldType == RelManyToMany {
|
||
|
ex = fi.relModelInfo.name
|
||
|
goto check
|
||
|
}
|
||
|
|
||
|
if fi.reverse && fi.reverseFieldInfo.fieldType == RelManyToMany {
|
||
|
ex = fi.reverseFieldInfo.mi.name
|
||
|
goto check
|
||
|
}
|
||
|
|
||
|
exist = true
|
||
|
|
||
|
} else {
|
||
|
|
||
|
if ffi == nil {
|
||
|
index = "T0"
|
||
|
} else {
|
||
|
index = jtl.index
|
||
|
}
|
||
|
column = fi.column
|
||
|
info = fi
|
||
|
if jtl != nil {
|
||
|
name = jtl.name + ExprSep + fi.name
|
||
|
} else {
|
||
|
name = fi.name
|
||
|
}
|
||
|
|
||
|
switch fi.fieldType {
|
||
|
case RelManyToMany, RelReverseMany:
|
||
|
default:
|
||
|
exist = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ffi = fi
|
||
|
}
|
||
|
|
||
|
if exist == false {
|
||
|
index = ""
|
||
|
column = ""
|
||
|
name = ""
|
||
|
success = false
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
success = index != "" && column != ""
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (d *dbTables) getCondSql(cond *Condition, sub bool) (where string, params []interface{}) {
|
||
|
if cond == nil || cond.IsEmpty() {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
Q := d.base.TableQuote()
|
||
|
|
||
|
mi := d.mi
|
||
|
|
||
|
// outFor:
|
||
|
for i, p := range cond.params {
|
||
|
if i > 0 {
|
||
|
if p.isOr {
|
||
|
where += "OR "
|
||
|
} else {
|
||
|
where += "AND "
|
||
|
}
|
||
|
}
|
||
|
if p.isNot {
|
||
|
where += "NOT "
|
||
|
}
|
||
|
if p.isCond {
|
||
|
w, ps := d.getCondSql(p.cond, true)
|
||
|
if w != "" {
|
||
|
w = fmt.Sprintf("( %s) ", w)
|
||
|
}
|
||
|
where += w
|
||
|
params = append(params, ps...)
|
||
|
} else {
|
||
|
exprs := p.exprs
|
||
|
|
||
|
num := len(exprs) - 1
|
||
|
operator := ""
|
||
|
if operators[exprs[num]] {
|
||
|
operator = exprs[num]
|
||
|
exprs = exprs[:num]
|
||
|
}
|
||
|
|
||
|
index, column, _, _, suc := d.parseExprs(mi, exprs)
|
||
|
if suc == false {
|
||
|
panic(fmt.Errorf("unknown field/column name `%s`", strings.Join(p.exprs, ExprSep)))
|
||
|
}
|
||
|
|
||
|
if operator == "" {
|
||
|
operator = "exact"
|
||
|
}
|
||
|
|
||
|
operSql, args := d.base.GenerateOperatorSql(mi, operator, p.args)
|
||
|
|
||
|
where += fmt.Sprintf("%s.%s%s%s %s ", index, Q, column, Q, operSql)
|
||
|
params = append(params, args...)
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if sub == false && where != "" {
|
||
|
where = "WHERE " + where
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (d *dbTables) getOrderSql(orders []string) (orderSql string) {
|
||
|
if len(orders) == 0 {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
Q := d.base.TableQuote()
|
||
|
|
||
|
orderSqls := make([]string, 0, len(orders))
|
||
|
for _, order := range orders {
|
||
|
asc := "ASC"
|
||
|
if order[0] == '-' {
|
||
|
asc = "DESC"
|
||
|
order = order[1:]
|
||
|
}
|
||
|
exprs := strings.Split(order, ExprSep)
|
||
|
|
||
|
index, column, _, _, suc := d.parseExprs(d.mi, exprs)
|
||
|
if suc == false {
|
||
|
panic(fmt.Errorf("unknown field/column name `%s`", strings.Join(exprs, ExprSep)))
|
||
|
}
|
||
|
|
||
|
orderSqls = append(orderSqls, fmt.Sprintf("%s.%s%s%s %s", index, Q, column, Q, asc))
|
||
|
}
|
||
|
|
||
|
orderSql = fmt.Sprintf("ORDER BY %s ", strings.Join(orderSqls, ", "))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (d *dbTables) getLimitSql(mi *modelInfo, offset int64, limit int) (limits string) {
|
||
|
if limit == 0 {
|
||
|
limit = DefaultRowsLimit
|
||
|
}
|
||
|
if limit < 0 {
|
||
|
// no limit
|
||
|
if offset > 0 {
|
||
|
maxLimit := d.base.MaxLimit()
|
||
|
limits = fmt.Sprintf("LIMIT %d OFFSET %d", maxLimit, offset)
|
||
|
}
|
||
|
} else if offset <= 0 {
|
||
|
limits = fmt.Sprintf("LIMIT %d", limit)
|
||
|
} else {
|
||
|
limits = fmt.Sprintf("LIMIT %d OFFSET %d", limit, offset)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func newDbTables(mi *modelInfo, base dbBaser) *dbTables {
|
||
|
tables := &dbTables{}
|
||
|
tables.tablesM = make(map[string]*dbTable)
|
||
|
tables.mi = mi
|
||
|
tables.base = base
|
||
|
return tables
|
||
|
}
|