Skip to content

Commit

Permalink
Test Joins
Browse files Browse the repository at this point in the history
  • Loading branch information
jinzhu committed May 31, 2020
1 parent 749ca37 commit 5b1d3e4
Show file tree
Hide file tree
Showing 4 changed files with 61 additions and 12 deletions.
6 changes: 1 addition & 5 deletions callbacks/query.go
Original file line number Diff line number Diff line change
Expand Up @@ -123,11 +123,7 @@ func BuildQuerySQL(db *gorm.DB) {
db.Statement.AddClauseIfNotExists(clause.From{})
}

if len(clauseSelect.Columns) > 0 {
db.Statement.AddClause(clauseSelect)
} else {
db.Statement.AddClauseIfNotExists(clauseSelect)
}
db.Statement.AddClauseIfNotExists(clauseSelect)

db.Statement.Build("SELECT", "FROM", "WHERE", "GROUP BY", "ORDER BY", "LIMIT", "FOR")
}
Expand Down
5 changes: 3 additions & 2 deletions finisher_api.go
Original file line number Diff line number Diff line change
Expand Up @@ -233,9 +233,10 @@ func (db *DB) Delete(value interface{}, conds ...interface{}) (tx *DB) {

func (db *DB) Count(count *int64) (tx *DB) {
tx = db.getInstance()
if len(tx.Statement.Selects) == 0 {
tx.Statement.Selects = []string{"count(1)"}
if s, ok := tx.Statement.Clauses["SELECT"].Expression.(clause.Select); !ok || len(s.Columns) == 0 {
tx.Statement.AddClause(clause.Select{Expression: clause.Expr{SQL: "count(1)"}})
}

if tx.Statement.Model == nil {
tx.Statement.Model = tx.Statement.Dest
}
Expand Down
10 changes: 5 additions & 5 deletions statement.go
Original file line number Diff line number Diff line change
Expand Up @@ -196,7 +196,7 @@ func (stmt *Statement) AddClause(v clause.Interface) {

// AddClauseIfNotExists add clause if not exists
func (stmt *Statement) AddClauseIfNotExists(v clause.Interface) {
if _, ok := stmt.Clauses[v.Name()]; !ok {
if c, ok := stmt.Clauses[v.Name()]; !ok && c.Expression == nil {
stmt.AddClause(v)
}
}
Expand Down Expand Up @@ -248,9 +248,9 @@ func (stmt Statement) BuildCondtion(query interface{}, args ...interface{}) (con
for _, field := range s.Fields {
if v, isZero := field.ValueOf(reflectValue); !isZero {
if field.DBName == "" {
conds = append(conds, clause.Eq{Column: field.Name, Value: v})
conds = append(conds, clause.Eq{Column: clause.Column{Table: s.Table, Name: field.Name}, Value: v})
} else {
conds = append(conds, clause.Eq{Column: field.DBName, Value: v})
conds = append(conds, clause.Eq{Column: clause.Column{Table: s.Table, Name: field.DBName}, Value: v})
}
}
}
Expand All @@ -259,9 +259,9 @@ func (stmt Statement) BuildCondtion(query interface{}, args ...interface{}) (con
for _, field := range s.Fields {
if v, isZero := field.ValueOf(reflectValue.Index(i)); !isZero {
if field.DBName == "" {
conds = append(conds, clause.Eq{Column: field.Name, Value: v})
conds = append(conds, clause.Eq{Column: clause.Column{Table: s.Table, Name: field.Name}, Value: v})
} else {
conds = append(conds, clause.Eq{Column: field.DBName, Value: v})
conds = append(conds, clause.Eq{Column: clause.Column{Table: s.Table, Name: field.DBName}, Value: v})
}
}
}
Expand Down
52 changes: 52 additions & 0 deletions tests/joins_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import (
"sort"
"testing"

"github.com/jinzhu/gorm"
. "github.com/jinzhu/gorm/tests"
)

Expand Down Expand Up @@ -53,3 +54,54 @@ func TestJoinsForSlice(t *testing.T) {
CheckUser(t, user, users2[idx])
}
}

func TestJoinConds(t *testing.T) {
var user = *GetUser("joins-conds", Config{Account: true, Pets: 3})
DB.Save(&user)

var users1 []User
DB.Joins("left join pets on pets.user_id = users.id").Where("users.name = ?", user.Name).Find(&users1)
if len(users1) != 3 {
t.Errorf("should find two users using left join, but got %v", len(users1))
}

var users2 []User
DB.Joins("left join pets on pets.user_id = users.id AND pets.name = ?", user.Pets[0].Name).Where("users.name = ?", user.Name).First(&users2)
if len(users2) != 1 {
t.Errorf("should find one users using left join with conditions, but got %v", len(users2))
}

var users3 []User
DB.Joins("left join pets on pets.user_id = users.id AND pets.name = ?", user.Pets[0].Name).Joins("join accounts on accounts.user_id = users.id AND accounts.number = ?", user.Account.Number).Where("users.name = ?", user.Name).First(&users3)
if len(users3) != 1 {
t.Errorf("should find one users using multiple left join conditions, but got %v", len(users3))
}

var users4 []User
DB.Joins("left join pets on pets.user_id = users.id AND pets.name = ?", user.Pets[0].Name).Joins("join accounts on accounts.user_id = users.id AND accounts.number = ?", user.Account.Number+"non-exist").Where("users.name = ?", user.Name).First(&users4)
if len(users4) != 0 {
t.Errorf("should find no user when searching with unexisting credit card, but got %v", len(users4))
}

var users5 []User
db5 := DB.Joins("left join pets on pets.user_id = users.id AND pets.name = ?", user.Pets[0].Name).Joins("join accounts on accounts.user_id = users.id AND accounts.number = ?", user.Account.Number).Where(User{Model: gorm.Model{ID: 1}}).Where(Account{Model: gorm.Model{ID: 1}}).Not(Pet{Model: gorm.Model{ID: 1}}).Find(&users5)
if db5.Error != nil {
t.Errorf("Should not raise error for join where identical fields in different tables. Error: %s", db5.Error.Error())
}
}

func TestJoinsWithSelect(t *testing.T) {
type result struct {
ID uint
Name string
}

user := *GetUser("joins_with_select", Config{Pets: 2})
DB.Save(&user)

var results []result
DB.Table("users").Select("users.id, pets.name").Joins("left join pets on pets.user_id = users.id").Where("users.name = ?", "joins_with_select").Scan(&results)
if len(results) != 2 || results[0].Name != user.Pets[0].Name || results[1].Name != user.Pets[1].Name {
t.Errorf("Should find all two pets with Join select")
}
}

0 comments on commit 5b1d3e4

Please sign in to comment.