gorm/association_test.go

276 lines
8.0 KiB
Go
Raw Normal View History

2014-07-29 08:32:58 +04:00
package gorm_test
import "testing"
2014-11-26 08:22:43 +03:00
import "github.com/jinzhu/gorm"
type Cat struct {
Id int
Name string
Toy Toy `gorm:"polymorphic:Owner;"`
}
type Dog struct {
Id int
Name string
Toys []Toy `gorm:"polymorphic:Owner;"`
}
type Toy struct {
Id int
Name string
OwnerId int
OwnerType string
// Define the owner type as a belongs_to so we can ensure it throws an error
Owner Dog `gorm:"foreignkey:owner_id; foreigntype:owner_type;"`
}
2014-07-29 08:32:58 +04:00
2014-07-30 10:58:00 +04:00
func TestHasOneAndHasManyAssociation(t *testing.T) {
2014-08-28 11:33:43 +04:00
DB.DropTable(Category{})
DB.DropTable(Post{})
DB.DropTable(Comment{})
2014-07-29 08:32:58 +04:00
2014-08-28 11:33:43 +04:00
DB.CreateTable(Category{})
DB.CreateTable(Post{})
DB.CreateTable(Comment{})
2014-07-29 08:32:58 +04:00
post := Post{
Title: "post 1",
Body: "body 1",
Comments: []Comment{{Content: "Comment 1"}, {Content: "Comment 2"}},
Category: Category{Name: "Category 1"},
MainCategory: Category{Name: "Main Category 1"},
}
2014-08-28 11:33:43 +04:00
if err := DB.Save(&post).Error; err != nil {
2014-07-29 08:32:58 +04:00
t.Errorf("Got errors when save post")
}
2014-08-28 11:33:43 +04:00
if DB.First(&Category{}, "name = ?", "Category 1").Error != nil {
2014-07-29 08:32:58 +04:00
t.Errorf("Category should be saved")
}
var p Post
2014-08-28 11:33:43 +04:00
DB.First(&p, post.Id)
2014-07-29 08:32:58 +04:00
if post.CategoryId.Int64 == 0 || p.CategoryId.Int64 == 0 || post.MainCategoryId == 0 || p.MainCategoryId == 0 {
t.Errorf("Category Id should exist")
}
2014-08-28 11:33:43 +04:00
if DB.First(&Comment{}, "content = ?", "Comment 1").Error != nil {
2014-07-29 08:32:58 +04:00
t.Errorf("Comment 1 should be saved")
}
if post.Comments[0].PostId == 0 {
t.Errorf("Comment Should have post id")
}
var comment Comment
2014-08-28 11:33:43 +04:00
if DB.First(&comment, "content = ?", "Comment 2").Error != nil {
2014-07-29 08:32:58 +04:00
t.Errorf("Comment 2 should be saved")
}
if comment.PostId == 0 {
t.Errorf("Comment 2 Should have post id")
}
comment3 := Comment{Content: "Comment 3", Post: Post{Title: "Title 3", Body: "Body 3"}}
2014-08-28 11:33:43 +04:00
DB.Save(&comment3)
2014-07-29 08:32:58 +04:00
}
func TestRelated(t *testing.T) {
user := User{
Name: "jinzhu",
BillingAddress: Address{Address1: "Billing Address - Address 1"},
ShippingAddress: Address{Address1: "Shipping Address - Address 1"},
Emails: []Email{{Email: "jinzhu@example.com"}, {Email: "jinzhu-2@example@example.com"}},
CreditCard: CreditCard{Number: "1234567890"},
}
2014-08-28 11:33:43 +04:00
DB.Save(&user)
2014-07-29 08:32:58 +04:00
if user.CreditCard.Id == 0 {
t.Errorf("After user save, credit card should have id")
}
if user.BillingAddress.Id == 0 {
t.Errorf("After user save, billing address should have id")
}
if user.Emails[0].Id == 0 {
t.Errorf("After user save, billing address should have id")
}
var emails []Email
2014-08-28 11:33:43 +04:00
DB.Model(&user).Related(&emails)
2014-07-29 08:32:58 +04:00
if len(emails) != 2 {
t.Errorf("Should have two emails")
}
var emails2 []Email
2014-08-28 11:33:43 +04:00
DB.Model(&user).Where("email = ?", "jinzhu@example.com").Related(&emails2)
2014-07-29 08:32:58 +04:00
if len(emails2) != 1 {
t.Errorf("Should have two emails")
}
var user1 User
2014-08-28 11:33:43 +04:00
DB.Model(&user).Related(&user1.Emails)
2014-07-29 08:32:58 +04:00
if len(user1.Emails) != 2 {
t.Errorf("Should have only one email match related condition")
}
var address1 Address
2014-08-28 11:33:43 +04:00
DB.Model(&user).Related(&address1, "BillingAddressId")
2014-07-29 08:32:58 +04:00
if address1.Address1 != "Billing Address - Address 1" {
t.Errorf("Should get billing address from user correctly")
}
user1 = User{}
2014-08-28 11:33:43 +04:00
DB.Model(&address1).Related(&user1, "BillingAddressId")
if DB.NewRecord(user1) {
2014-07-29 08:32:58 +04:00
t.Errorf("Should get user from address correctly")
}
var user2 User
2014-08-28 11:33:43 +04:00
DB.Model(&emails[0]).Related(&user2)
2014-07-29 08:32:58 +04:00
if user2.Id != user.Id || user2.Name != user.Name {
t.Errorf("Should get user from email correctly")
}
var creditcard CreditCard
var user3 User
2014-08-28 11:33:43 +04:00
DB.First(&creditcard, "number = ?", "1234567890")
DB.Model(&creditcard).Related(&user3)
2014-07-29 08:32:58 +04:00
if user3.Id != user.Id || user3.Name != user.Name {
t.Errorf("Should get user from credit card correctly")
}
2014-08-28 11:33:43 +04:00
if !DB.Model(&CreditCard{}).Related(&User{}).RecordNotFound() {
2014-07-29 08:32:58 +04:00
t.Errorf("RecordNotFound for Related")
}
}
2014-07-30 10:58:00 +04:00
func TestManyToMany(t *testing.T) {
2014-08-28 11:33:43 +04:00
DB.Raw("delete from languages")
2014-07-30 16:48:36 +04:00
var languages = []Language{{Name: "ZH"}, {Name: "EN"}}
2014-07-29 13:28:10 +04:00
user := User{Name: "Many2Many", Languages: languages}
2014-08-28 11:33:43 +04:00
DB.Save(&user)
2014-07-29 13:28:10 +04:00
2014-07-30 16:48:36 +04:00
// Query
2014-07-30 07:32:18 +04:00
var newLanguages []Language
2014-08-28 11:33:43 +04:00
DB.Model(&user).Related(&newLanguages, "Languages")
2014-07-30 16:48:36 +04:00
if len(newLanguages) != len([]string{"ZH", "EN"}) {
t.Errorf("Query many to many relations")
}
2014-07-30 07:32:18 +04:00
newLanguages = []Language{}
2014-08-28 11:33:43 +04:00
DB.Model(&user).Association("Languages").Find(&newLanguages)
2014-07-30 16:48:36 +04:00
if len(newLanguages) != len([]string{"ZH", "EN"}) {
2014-07-30 12:22:26 +04:00
t.Errorf("Should be able to find many to many relations")
2014-07-30 07:32:18 +04:00
}
2014-07-29 13:28:10 +04:00
2014-08-28 11:33:43 +04:00
if DB.Model(&user).Association("Languages").Count() != len([]string{"ZH", "EN"}) {
2014-07-30 17:43:53 +04:00
t.Errorf("Count should return correct result")
}
2014-07-30 16:48:36 +04:00
// Append
2014-08-28 11:33:43 +04:00
DB.Model(&user).Association("Languages").Append(&Language{Name: "DE"})
if DB.Where("name = ?", "DE").First(&Language{}).RecordNotFound() {
2014-07-30 16:48:36 +04:00
t.Errorf("New record should be saved when append")
}
languageA := Language{Name: "AA"}
2014-08-28 11:33:43 +04:00
DB.Save(&languageA)
DB.Model(&User{Id: user.Id}).Association("Languages").Append(languageA)
2014-09-02 15:03:01 +04:00
2014-07-30 16:48:36 +04:00
languageC := Language{Name: "CC"}
2014-08-28 11:33:43 +04:00
DB.Save(&languageC)
DB.Model(&user).Association("Languages").Append(&[]Language{{Name: "BB"}, languageC})
2014-09-02 15:03:01 +04:00
DB.Model(&User{Id: user.Id}).Association("Languages").Append(&[]Language{{Name: "DD"}, {Name: "EE"}})
2014-07-30 16:48:36 +04:00
totalLanguages := []string{"ZH", "EN", "DE", "AA", "BB", "CC", "DD", "EE"}
2014-08-28 11:33:43 +04:00
if DB.Model(&user).Association("Languages").Count() != len(totalLanguages) {
2014-07-30 16:48:36 +04:00
t.Errorf("All appended languages should be saved")
}
// Delete
var language Language
2014-08-28 11:33:43 +04:00
DB.Where("name = ?", "EE").First(&language)
DB.Model(&user).Association("Languages").Delete(language, &language)
if DB.Model(&user).Association("Languages").Count() != len(totalLanguages)-1 {
2014-07-30 16:48:36 +04:00
t.Errorf("Relations should be deleted with Delete")
}
2014-08-28 11:33:43 +04:00
if DB.Where("name = ?", "EE").First(&Language{}).RecordNotFound() {
2014-07-30 16:48:36 +04:00
t.Errorf("Language EE should not be deleted")
}
languages = []Language{}
2014-08-28 11:33:43 +04:00
DB.Where("name IN (?)", []string{"CC", "DD"}).Find(&languages)
DB.Model(&user).Association("Languages").Delete(languages, &languages)
if DB.Model(&user).Association("Languages").Count() != len(totalLanguages)-3 {
2014-07-30 16:48:36 +04:00
t.Errorf("Relations should be deleted with Delete")
}
2014-07-30 18:10:12 +04:00
// Replace
var languageB Language
2014-08-28 11:33:43 +04:00
DB.Where("name = ?", "BB").First(&languageB)
DB.Model(&user).Association("Languages").Replace(languageB)
if DB.Model(&user).Association("Languages").Count() != 1 {
2014-07-30 18:15:31 +04:00
t.Errorf("Relations should be replaced")
2014-07-30 18:10:12 +04:00
}
2014-08-28 11:33:43 +04:00
DB.Model(&user).Association("Languages").Replace(&[]Language{{Name: "FF"}, {Name: "JJ"}})
if DB.Model(&user).Association("Languages").Count() != len([]string{"FF", "JJ"}) {
2014-07-30 18:15:31 +04:00
t.Errorf("Relations should be replaced")
2014-07-30 18:10:12 +04:00
}
2014-07-30 18:15:31 +04:00
// Clear
2014-08-28 11:33:43 +04:00
DB.Model(&user).Association("Languages").Clear()
if DB.Model(&user).Association("Languages").Count() != 0 {
2014-07-30 18:15:31 +04:00
t.Errorf("Relations should be cleared")
}
2014-07-29 08:32:58 +04:00
}
2014-11-26 08:22:43 +03:00
func TestPolymorphic(t *testing.T) {
DB.DropTableIfExists(Cat{})
DB.DropTableIfExists(Dog{})
DB.DropTableIfExists(Toy{})
DB.AutoMigrate(&Cat{})
DB.AutoMigrate(&Dog{})
DB.AutoMigrate(&Toy{})
cat := Cat{Name: "Mr. Bigglesworth", Toy: Toy{Name: "cat nip"}}
dog := Dog{Name: "Pluto", Toys: []Toy{Toy{Name: "orange ball"}, Toy{Name: "yellow ball"}}}
DB.Save(&cat).Save(&dog)
var catToys []Toy
if err := DB.Model(&cat).Related(&catToys, "Toy").Error; err == gorm.RecordNotFound {
t.Errorf("Did not find any has one polymorphic association")
} else if len(catToys) != 1 {
t.Errorf("Should have found only one polymorphic has one association")
} else if catToys[0].Name != cat.Toy.Name {
t.Errorf("Should have found the proper has one polymorphic association")
}
var dogToys []Toy
if err := DB.Model(&dog).Related(&dogToys, "Toys").Error; err == gorm.RecordNotFound {
t.Errorf("Did not find any polymorphic has many associations")
} else if len(dogToys) != len(dog.Toys) {
t.Errorf("Should have found all polymorphic has many associations")
}
if DB.Model(&cat).Association("Toy").Count() != 1 {
t.Errorf("Should return one polymorphic has one association")
}
if DB.Model(&dog).Association("Toys").Count() != 2 {
t.Errorf("Should return two polymorphic has many associations")
}
if DB.Model(&Toy{OwnerId: dog.Id, OwnerType: "dog"}).Related(&dog, "Owner").Error == nil {
t.Errorf("Should have thrown unsupported belongs_to error")
}
}