query.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. package callbacks
  2. import (
  3. "fmt"
  4. "reflect"
  5. "sort"
  6. "strings"
  7. "gorm.io/gorm"
  8. "gorm.io/gorm/clause"
  9. "gorm.io/gorm/schema"
  10. )
  11. func Query(db *gorm.DB) {
  12. if db.Error == nil {
  13. BuildQuerySQL(db)
  14. if !db.DryRun && db.Error == nil {
  15. rows, err := db.Statement.ConnPool.QueryContext(db.Statement.Context, db.Statement.SQL.String(), db.Statement.Vars...)
  16. if err != nil {
  17. db.AddError(err)
  18. return
  19. }
  20. defer rows.Close()
  21. gorm.Scan(rows, db, false)
  22. }
  23. }
  24. }
  25. func BuildQuerySQL(db *gorm.DB) {
  26. if db.Statement.Schema != nil && !db.Statement.Unscoped {
  27. for _, c := range db.Statement.Schema.QueryClauses {
  28. db.Statement.AddClause(c)
  29. }
  30. }
  31. if db.Statement.SQL.String() == "" {
  32. db.Statement.SQL.Grow(100)
  33. clauseSelect := clause.Select{Distinct: db.Statement.Distinct}
  34. if db.Statement.ReflectValue.Kind() == reflect.Struct && db.Statement.ReflectValue.Type() == db.Statement.Schema.ModelType {
  35. var conds []clause.Expression
  36. for _, primaryField := range db.Statement.Schema.PrimaryFields {
  37. if v, isZero := primaryField.ValueOf(db.Statement.ReflectValue); !isZero {
  38. conds = append(conds, clause.Eq{Column: clause.Column{Table: db.Statement.Table, Name: primaryField.DBName}, Value: v})
  39. }
  40. }
  41. if len(conds) > 0 {
  42. db.Statement.AddClause(clause.Where{Exprs: conds})
  43. }
  44. }
  45. if len(db.Statement.Selects) > 0 {
  46. clauseSelect.Columns = make([]clause.Column, len(db.Statement.Selects))
  47. for idx, name := range db.Statement.Selects {
  48. if db.Statement.Schema == nil {
  49. clauseSelect.Columns[idx] = clause.Column{Name: name, Raw: true}
  50. } else if f := db.Statement.Schema.LookUpField(name); f != nil {
  51. clauseSelect.Columns[idx] = clause.Column{Name: f.DBName}
  52. } else {
  53. clauseSelect.Columns[idx] = clause.Column{Name: name, Raw: true}
  54. }
  55. }
  56. } else if db.Statement.Schema != nil && len(db.Statement.Omits) > 0 {
  57. selectColumns, _ := db.Statement.SelectAndOmitColumns(false, false)
  58. clauseSelect.Columns = make([]clause.Column, 0, len(db.Statement.Schema.DBNames))
  59. for _, dbName := range db.Statement.Schema.DBNames {
  60. if v, ok := selectColumns[dbName]; (ok && v) || !ok {
  61. clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{Name: dbName})
  62. }
  63. }
  64. } else if db.Statement.Schema != nil && db.Statement.ReflectValue.IsValid() {
  65. smallerStruct := false
  66. switch db.Statement.ReflectValue.Kind() {
  67. case reflect.Struct:
  68. smallerStruct = db.Statement.ReflectValue.Type() != db.Statement.Schema.ModelType
  69. case reflect.Slice:
  70. smallerStruct = db.Statement.ReflectValue.Type().Elem() != db.Statement.Schema.ModelType
  71. }
  72. if smallerStruct {
  73. stmt := gorm.Statement{DB: db}
  74. // smaller struct
  75. if err := stmt.Parse(db.Statement.Dest); err == nil && stmt.Schema.ModelType != db.Statement.Schema.ModelType {
  76. clauseSelect.Columns = make([]clause.Column, len(stmt.Schema.DBNames))
  77. for idx, dbName := range stmt.Schema.DBNames {
  78. clauseSelect.Columns[idx] = clause.Column{Name: dbName}
  79. }
  80. }
  81. }
  82. }
  83. // inline joins
  84. if len(db.Statement.Joins) != 0 {
  85. if len(db.Statement.Selects) == 0 && db.Statement.Schema != nil {
  86. clauseSelect.Columns = make([]clause.Column, len(db.Statement.Schema.DBNames))
  87. for idx, dbName := range db.Statement.Schema.DBNames {
  88. clauseSelect.Columns[idx] = clause.Column{Table: db.Statement.Table, Name: dbName}
  89. }
  90. }
  91. joins := []clause.Join{}
  92. for _, join := range db.Statement.Joins {
  93. if db.Statement.Schema == nil {
  94. joins = append(joins, clause.Join{
  95. Expression: clause.Expr{SQL: join.Name, Vars: join.Conds},
  96. })
  97. } else if relation, ok := db.Statement.Schema.Relationships.Relations[join.Name]; ok {
  98. tableAliasName := relation.Name
  99. for _, s := range relation.FieldSchema.DBNames {
  100. clauseSelect.Columns = append(clauseSelect.Columns, clause.Column{
  101. Table: tableAliasName,
  102. Name: s,
  103. Alias: tableAliasName + "__" + s,
  104. })
  105. }
  106. exprs := make([]clause.Expression, len(relation.References))
  107. for idx, ref := range relation.References {
  108. if ref.OwnPrimaryKey {
  109. exprs[idx] = clause.Eq{
  110. Column: clause.Column{Table: clause.CurrentTable, Name: ref.PrimaryKey.DBName},
  111. Value: clause.Column{Table: tableAliasName, Name: ref.ForeignKey.DBName},
  112. }
  113. } else {
  114. if ref.PrimaryValue == "" {
  115. exprs[idx] = clause.Eq{
  116. Column: clause.Column{Table: clause.CurrentTable, Name: ref.ForeignKey.DBName},
  117. Value: clause.Column{Table: tableAliasName, Name: ref.PrimaryKey.DBName},
  118. }
  119. } else {
  120. exprs[idx] = clause.Eq{
  121. Column: clause.Column{Table: tableAliasName, Name: ref.ForeignKey.DBName},
  122. Value: ref.PrimaryValue,
  123. }
  124. }
  125. }
  126. }
  127. joins = append(joins, clause.Join{
  128. Type: clause.LeftJoin,
  129. Table: clause.Table{Name: relation.FieldSchema.Table, Alias: tableAliasName},
  130. ON: clause.Where{Exprs: exprs},
  131. })
  132. } else {
  133. joins = append(joins, clause.Join{
  134. Expression: clause.Expr{SQL: join.Name, Vars: join.Conds},
  135. })
  136. }
  137. }
  138. db.Statement.AddClause(clause.From{Joins: joins})
  139. } else {
  140. db.Statement.AddClauseIfNotExists(clause.From{})
  141. }
  142. db.Statement.AddClauseIfNotExists(clauseSelect)
  143. db.Statement.Build("SELECT", "FROM", "WHERE", "GROUP BY", "ORDER BY", "LIMIT", "FOR")
  144. }
  145. }
  146. func Preload(db *gorm.DB) {
  147. if db.Error == nil && len(db.Statement.Preloads) > 0 {
  148. preloadMap := map[string][]string{}
  149. for name := range db.Statement.Preloads {
  150. if name == clause.Associations {
  151. for _, rel := range db.Statement.Schema.Relationships.Relations {
  152. if rel.Schema == db.Statement.Schema {
  153. preloadMap[rel.Name] = []string{rel.Name}
  154. }
  155. }
  156. } else {
  157. preloadFields := strings.Split(name, ".")
  158. for idx := range preloadFields {
  159. preloadMap[strings.Join(preloadFields[:idx+1], ".")] = preloadFields[:idx+1]
  160. }
  161. }
  162. }
  163. preloadNames := make([]string, len(preloadMap))
  164. idx := 0
  165. for key := range preloadMap {
  166. preloadNames[idx] = key
  167. idx++
  168. }
  169. sort.Strings(preloadNames)
  170. for _, name := range preloadNames {
  171. var (
  172. curSchema = db.Statement.Schema
  173. preloadFields = preloadMap[name]
  174. rels = make([]*schema.Relationship, len(preloadFields))
  175. )
  176. for idx, preloadField := range preloadFields {
  177. if rel := curSchema.Relationships.Relations[preloadField]; rel != nil {
  178. rels[idx] = rel
  179. curSchema = rel.FieldSchema
  180. } else {
  181. db.AddError(fmt.Errorf("%v: %w", name, gorm.ErrUnsupportedRelation))
  182. }
  183. }
  184. preload(db, rels, db.Statement.Preloads[name])
  185. }
  186. }
  187. }
  188. func AfterQuery(db *gorm.DB) {
  189. if db.Error == nil && db.Statement.Schema != nil && db.Statement.Schema.AfterFind {
  190. callMethod(db, func(value interface{}, tx *gorm.DB) bool {
  191. if i, ok := value.(AfterFindInterface); ok {
  192. db.AddError(i.AfterFind(tx))
  193. return true
  194. }
  195. return false
  196. })
  197. }
  198. }