2020-02-02 09:40:44 +03:00
|
|
|
package callbacks
|
|
|
|
|
2020-02-04 03:56:15 +03:00
|
|
|
import (
|
2020-04-15 04:14:24 +03:00
|
|
|
"fmt"
|
2020-02-23 16:22:35 +03:00
|
|
|
"reflect"
|
2020-05-07 05:03:48 +03:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2020-02-23 16:22:35 +03:00
|
|
|
|
2020-06-02 04:16:07 +03:00
|
|
|
"gorm.io/gorm"
|
|
|
|
"gorm.io/gorm/clause"
|
|
|
|
"gorm.io/gorm/schema"
|
2020-02-04 03:56:15 +03:00
|
|
|
)
|
2020-02-02 09:40:44 +03:00
|
|
|
|
|
|
|
func Query(db *gorm.DB) {
|
2020-05-31 18:55:56 +03:00
|
|
|
if db.Error == nil {
|
|
|
|
if db.Statement.Schema != nil && !db.Statement.Unscoped {
|
|
|
|
for _, c := range db.Statement.Schema.QueryClauses {
|
|
|
|
db.Statement.AddClause(c)
|
|
|
|
}
|
2020-05-29 02:35:45 +03:00
|
|
|
}
|
|
|
|
|
2020-05-31 18:55:56 +03:00
|
|
|
if db.Statement.SQL.String() == "" {
|
|
|
|
BuildQuerySQL(db)
|
|
|
|
}
|
2020-04-15 04:14:24 +03:00
|
|
|
|
2020-06-01 16:26:23 +03:00
|
|
|
if !db.DryRun {
|
|
|
|
rows, err := db.Statement.ConnPool.QueryContext(db.Statement.Context, db.Statement.SQL.String(), db.Statement.Vars...)
|
|
|
|
if err != nil {
|
|
|
|
db.AddError(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
2020-05-31 14:23:32 +03:00
|
|
|
|
2020-06-01 16:26:23 +03:00
|
|
|
gorm.Scan(rows, db, false)
|
|
|
|
}
|
2020-05-31 18:55:56 +03:00
|
|
|
}
|
2020-05-31 14:23:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func BuildQuerySQL(db *gorm.DB) {
|
|
|
|
clauseSelect := clause.Select{}
|
|
|
|
|
2020-06-01 14:41:33 +03:00
|
|
|
if db.Statement.ReflectValue.Kind() == reflect.Struct {
|
|
|
|
var conds []clause.Expression
|
|
|
|
for _, primaryField := range db.Statement.Schema.PrimaryFields {
|
|
|
|
if v, isZero := primaryField.ValueOf(db.Statement.ReflectValue); !isZero {
|
|
|
|
conds = append(conds, clause.Eq{Column: clause.Column{Table: db.Statement.Table, Name: primaryField.DBName}, Value: v})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(conds) > 0 {
|
|
|
|
db.Statement.AddClause(clause.Where{Exprs: conds})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
if len(db.Statement.Selects) > 0 {
|
|
|
|
for _, name := range db.Statement.Selects {
|
|
|
|
if db.Statement.Schema == nil {
|
|
|
|
clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
|
|
|
|
Name: name,
|
|
|
|
Raw: true,
|
|
|
|
})
|
|
|
|
} else if f := db.Statement.Schema.LookUpField(name); f != nil {
|
|
|
|
clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
|
|
|
|
Name: f.DBName,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
|
|
|
|
Name: name,
|
|
|
|
Raw: true,
|
|
|
|
})
|
2020-04-15 04:14:24 +03:00
|
|
|
}
|
|
|
|
}
|
2020-05-31 14:23:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// inline joins
|
|
|
|
if len(db.Statement.Joins) != 0 {
|
|
|
|
joins := []clause.Join{}
|
2020-04-15 04:14:24 +03:00
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
if len(db.Statement.Selects) == 0 {
|
|
|
|
for _, dbName := range db.Statement.Schema.DBNames {
|
|
|
|
clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
|
|
|
|
Table: db.Statement.Table,
|
|
|
|
Name: dbName,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-04-15 04:14:24 +03:00
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
for name, conds := range db.Statement.Joins {
|
|
|
|
if db.Statement.Schema == nil {
|
|
|
|
joins = append(joins, clause.Join{
|
|
|
|
Expression: clause.Expr{SQL: name, Vars: conds},
|
|
|
|
})
|
|
|
|
} else if relation, ok := db.Statement.Schema.Relationships.Relations[name]; ok {
|
|
|
|
tableAliasName := relation.Name
|
|
|
|
|
|
|
|
for _, s := range relation.FieldSchema.DBNames {
|
2020-04-15 04:14:24 +03:00
|
|
|
clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
|
2020-05-31 14:23:32 +03:00
|
|
|
Table: tableAliasName,
|
|
|
|
Name: s,
|
|
|
|
Alias: tableAliasName + "__" + s,
|
2020-04-15 04:14:24 +03:00
|
|
|
})
|
|
|
|
}
|
2020-05-29 19:16:33 +03:00
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
var exprs []clause.Expression
|
|
|
|
for _, ref := range relation.References {
|
|
|
|
if ref.OwnPrimaryKey {
|
|
|
|
exprs = append(exprs, clause.Eq{
|
|
|
|
Column: clause.Column{Table: db.Statement.Schema.Table, Name: ref.PrimaryKey.DBName},
|
|
|
|
Value: clause.Column{Table: tableAliasName, Name: ref.ForeignKey.DBName},
|
2020-04-15 04:14:24 +03:00
|
|
|
})
|
2020-05-31 14:23:32 +03:00
|
|
|
} else {
|
|
|
|
if ref.PrimaryValue == "" {
|
2020-05-29 19:16:33 +03:00
|
|
|
exprs = append(exprs, clause.Eq{
|
2020-05-31 14:23:32 +03:00
|
|
|
Column: clause.Column{Table: db.Statement.Schema.Table, Name: ref.ForeignKey.DBName},
|
|
|
|
Value: clause.Column{Table: tableAliasName, Name: ref.PrimaryKey.DBName},
|
2020-04-15 04:14:24 +03:00
|
|
|
})
|
|
|
|
} else {
|
2020-05-31 14:23:32 +03:00
|
|
|
exprs = append(exprs, clause.Eq{
|
|
|
|
Column: clause.Column{Table: tableAliasName, Name: ref.ForeignKey.DBName},
|
|
|
|
Value: ref.PrimaryValue,
|
|
|
|
})
|
2020-04-15 04:14:24 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
joins = append(joins, clause.Join{
|
|
|
|
Type: clause.LeftJoin,
|
|
|
|
Table: clause.Table{Name: relation.FieldSchema.Table, Alias: tableAliasName},
|
|
|
|
ON: clause.Where{Exprs: exprs},
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
joins = append(joins, clause.Join{
|
|
|
|
Expression: clause.Expr{SQL: name, Vars: conds},
|
|
|
|
})
|
|
|
|
}
|
2020-04-15 04:14:24 +03:00
|
|
|
}
|
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
db.Statement.AddClause(clause.From{Joins: joins})
|
|
|
|
} else {
|
|
|
|
db.Statement.AddClauseIfNotExists(clause.From{})
|
2020-02-22 15:57:29 +03:00
|
|
|
}
|
2020-02-04 03:56:15 +03:00
|
|
|
|
2020-05-31 15:21:52 +03:00
|
|
|
db.Statement.AddClauseIfNotExists(clauseSelect)
|
2020-03-03 09:18:12 +03:00
|
|
|
|
2020-05-31 14:23:32 +03:00
|
|
|
db.Statement.Build("SELECT", "FROM", "WHERE", "GROUP BY", "ORDER BY", "LIMIT", "FOR")
|
2020-02-02 09:40:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func Preload(db *gorm.DB) {
|
2020-05-31 18:55:56 +03:00
|
|
|
if db.Error == nil {
|
|
|
|
if len(db.Statement.Preloads) > 0 {
|
|
|
|
preloadMap := map[string][]string{}
|
|
|
|
for name := range db.Statement.Preloads {
|
|
|
|
preloadFields := strings.Split(name, ".")
|
|
|
|
for idx := range preloadFields {
|
|
|
|
preloadMap[strings.Join(preloadFields[:idx+1], ".")] = preloadFields[:idx+1]
|
|
|
|
}
|
2020-05-07 05:03:48 +03:00
|
|
|
}
|
|
|
|
|
2020-05-31 18:55:56 +03:00
|
|
|
preloadNames := make([]string, len(preloadMap))
|
|
|
|
idx := 0
|
|
|
|
for key := range preloadMap {
|
|
|
|
preloadNames[idx] = key
|
|
|
|
idx++
|
2020-05-07 05:03:48 +03:00
|
|
|
}
|
2020-05-31 18:55:56 +03:00
|
|
|
sort.Strings(preloadNames)
|
|
|
|
|
|
|
|
for _, name := range preloadNames {
|
|
|
|
var (
|
|
|
|
curSchema = db.Statement.Schema
|
|
|
|
preloadFields = preloadMap[name]
|
|
|
|
rels = make([]*schema.Relationship, len(preloadFields))
|
|
|
|
)
|
|
|
|
|
|
|
|
for idx, preloadField := range preloadFields {
|
|
|
|
if rel := curSchema.Relationships.Relations[preloadField]; rel != nil {
|
|
|
|
rels[idx] = rel
|
|
|
|
curSchema = rel.FieldSchema
|
|
|
|
} else {
|
|
|
|
db.AddError(fmt.Errorf("%v: %w", name, gorm.ErrUnsupportedRelation))
|
|
|
|
}
|
|
|
|
}
|
2020-05-14 07:19:12 +03:00
|
|
|
|
2020-05-31 18:55:56 +03:00
|
|
|
preload(db, rels, db.Statement.Preloads[name])
|
|
|
|
}
|
2020-05-07 05:03:48 +03:00
|
|
|
}
|
|
|
|
}
|
2020-02-02 09:40:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func AfterQuery(db *gorm.DB) {
|
2020-05-31 18:55:56 +03:00
|
|
|
if db.Error == nil && db.Statement.Schema != nil && db.Statement.Schema.AfterFind {
|
|
|
|
tx := db.Session(&gorm.Session{})
|
2020-02-23 16:22:35 +03:00
|
|
|
callMethod := func(value interface{}) bool {
|
|
|
|
if db.Statement.Schema.AfterFind {
|
|
|
|
if i, ok := value.(gorm.AfterFindInterface); ok {
|
2020-05-31 18:55:56 +03:00
|
|
|
db.AddError(i.AfterFind(tx))
|
2020-02-23 16:22:35 +03:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if ok := callMethod(db.Statement.Dest); !ok {
|
|
|
|
switch db.Statement.ReflectValue.Kind() {
|
|
|
|
case reflect.Slice, reflect.Array:
|
2020-05-31 18:55:56 +03:00
|
|
|
for i := 0; i < db.Statement.ReflectValue.Len(); i++ {
|
2020-02-23 16:22:35 +03:00
|
|
|
callMethod(db.Statement.ReflectValue.Index(i).Interface())
|
|
|
|
}
|
|
|
|
case reflect.Struct:
|
|
|
|
callMethod(db.Statement.ReflectValue.Interface())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-02 09:40:44 +03:00
|
|
|
}
|