Test Many2Many Association for Slice
This commit is contained in:
		
							parent
							
								
									dea48a8c59
								
							
						
					
					
						commit
						457f1e5d73
					
				| @ -340,11 +340,16 @@ func (association *Association) Count() (count int64) { | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| type assignBack struct { | ||||
| 	Source reflect.Value | ||||
| 	Index  int | ||||
| 	Dest   reflect.Value | ||||
| } | ||||
| 
 | ||||
| func (association *Association) saveAssociation(clear bool, values ...interface{}) { | ||||
| 	var ( | ||||
| 		reflectValue = association.DB.Statement.ReflectValue | ||||
| 		assignBacks  = [][2]reflect.Value{} | ||||
| 		assignBack   = association.Relationship.Field.FieldType.Kind() == reflect.Struct | ||||
| 		assignBacks  []assignBack | ||||
| 	) | ||||
| 
 | ||||
| 	appendToRelations := func(source, rv reflect.Value, clear bool) { | ||||
| @ -354,14 +359,14 @@ func (association *Association) saveAssociation(clear bool, values ...interface{ | ||||
| 			case reflect.Slice, reflect.Array: | ||||
| 				if rv.Len() > 0 { | ||||
| 					association.Error = association.Relationship.Field.Set(source, rv.Index(0).Addr().Interface()) | ||||
| 					if assignBack { | ||||
| 						assignBacks = append(assignBacks, [2]reflect.Value{source, rv.Index(0)}) | ||||
| 					if association.Relationship.Field.FieldType.Kind() == reflect.Struct { | ||||
| 						assignBacks = append(assignBacks, assignBack{Source: source, Dest: rv.Index(0)}) | ||||
| 					} | ||||
| 				} | ||||
| 			case reflect.Struct: | ||||
| 				association.Error = association.Relationship.Field.Set(source, rv.Addr().Interface()) | ||||
| 				if assignBack { | ||||
| 					assignBacks = append(assignBacks, [2]reflect.Value{source, rv}) | ||||
| 				if association.Relationship.Field.FieldType.Kind() == reflect.Struct { | ||||
| 					assignBacks = append(assignBacks, assignBack{Source: source, Dest: rv}) | ||||
| 				} | ||||
| 			} | ||||
| 		case schema.HasMany, schema.Many2Many: | ||||
| @ -379,6 +384,14 @@ func (association *Association) saveAssociation(clear bool, values ...interface{ | ||||
| 				} else { | ||||
| 					association.Error = fmt.Errorf("unsupported data type: %v for relation %v", ev.Type(), association.Relationship.Name) | ||||
| 				} | ||||
| 
 | ||||
| 				if association.Relationship.Field.IndirectFieldType.Elem().Kind() == reflect.Struct { | ||||
| 					assignBacks = append(assignBacks, assignBack{ | ||||
| 						Source: source, | ||||
| 						Index:  fieldValue.Len(), | ||||
| 						Dest:   ev, | ||||
| 					}) | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			switch rv.Kind() { | ||||
| @ -451,6 +464,11 @@ func (association *Association) saveAssociation(clear bool, values ...interface{ | ||||
| 	} | ||||
| 
 | ||||
| 	for _, assignBack := range assignBacks { | ||||
| 		reflect.Indirect(assignBack[1]).Set(association.Relationship.Field.ReflectValueOf(assignBack[0])) | ||||
| 		fieldValue := reflect.Indirect(association.Relationship.Field.ReflectValueOf(assignBack.Source)) | ||||
| 		if assignBack.Index > 0 { | ||||
| 			reflect.Indirect(assignBack.Dest).Set(fieldValue.Index(assignBack.Index - 1)) | ||||
| 		} else { | ||||
| 			reflect.Indirect(assignBack.Dest).Set(fieldValue) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -786,7 +786,7 @@ func TestMany2ManyAssociation(t *testing.T) { | ||||
| 	AssertAssociationCount(t, user, "Languages", 2, "") | ||||
| 
 | ||||
| 	// Append
 | ||||
| 	var language = Language{Code: "language-has-many-append", Name: "language-has-many-append"} | ||||
| 	var language = Language{Code: "language-many2many-append", Name: "language-many2many-append"} | ||||
| 	DB.Create(&language) | ||||
| 
 | ||||
| 	if err := DB.Model(&user2).Association("Languages").Append(&language); err != nil { | ||||
| @ -799,8 +799,8 @@ func TestMany2ManyAssociation(t *testing.T) { | ||||
| 	AssertAssociationCount(t, user, "Languages", 3, "AfterAppend") | ||||
| 
 | ||||
| 	var languages = []Language{ | ||||
| 		{Code: "language-has-many-append-1-1", Name: "language-has-many-append-1-1"}, | ||||
| 		{Code: "language-has-many-append-2-1", Name: "language-has-many-append-2-1"}, | ||||
| 		{Code: "language-many2many-append-1-1", Name: "language-many2many-append-1-1"}, | ||||
| 		{Code: "language-many2many-append-2-1", Name: "language-many2many-append-2-1"}, | ||||
| 	} | ||||
| 	DB.Create(&languages) | ||||
| 
 | ||||
| @ -815,7 +815,7 @@ func TestMany2ManyAssociation(t *testing.T) { | ||||
| 	AssertAssociationCount(t, user, "Languages", 5, "AfterAppendSlice") | ||||
| 
 | ||||
| 	// Replace
 | ||||
| 	var language2 = Language{Code: "language-has-many-replace", Name: "language-has-many-replace"} | ||||
| 	var language2 = Language{Code: "language-many2many-replace", Name: "language-many2many-replace"} | ||||
| 	DB.Create(&language2) | ||||
| 
 | ||||
| 	if err := DB.Model(&user2).Association("Languages").Replace(&language2); err != nil { | ||||
| @ -852,3 +852,73 @@ func TestMany2ManyAssociation(t *testing.T) { | ||||
| 
 | ||||
| 	AssertAssociationCount(t, user2, "Languages", 0, "after clear") | ||||
| } | ||||
| 
 | ||||
| func TestMany2ManyAssociationForSlice(t *testing.T) { | ||||
| 	var users = []User{ | ||||
| 		*GetUser("slice-many2many-1", Config{Languages: 2}), | ||||
| 		*GetUser("slice-many2many-2", Config{Languages: 0}), | ||||
| 		*GetUser("slice-many2many-3", Config{Languages: 4}), | ||||
| 	} | ||||
| 
 | ||||
| 	DB.Create(&users) | ||||
| 
 | ||||
| 	// Count
 | ||||
| 	AssertAssociationCount(t, users, "Languages", 6, "") | ||||
| 
 | ||||
| 	// Find
 | ||||
| 	var languages []Language | ||||
| 	if DB.Model(&users).Association("Languages").Find(&languages); len(languages) != 6 { | ||||
| 		t.Errorf("languages count should be %v, but got %v", 6, len(languages)) | ||||
| 	} | ||||
| 
 | ||||
| 	// Append
 | ||||
| 	var languages1 = []Language{ | ||||
| 		{Code: "language-many2many-append-1", Name: "language-many2many-append-1"}, | ||||
| 	} | ||||
| 	var languages2 = []Language{} | ||||
| 	var languages3 = []Language{ | ||||
| 		{Code: "language-many2many-append-3-1", Name: "language-many2many-append-3-1"}, | ||||
| 		{Code: "language-many2many-append-3-2", Name: "language-many2many-append-3-2"}, | ||||
| 	} | ||||
| 	DB.Create(&languages1) | ||||
| 	DB.Create(&languages3) | ||||
| 
 | ||||
| 	DB.Model(&users).Association("Languages").Append(&languages1, &languages2, &languages3) | ||||
| 
 | ||||
| 	AssertAssociationCount(t, users, "Languages", 9, "After Append") | ||||
| 
 | ||||
| 	languages2_1 := []*Language{ | ||||
| 		{Code: "language-slice-replace-1-1", Name: "language-slice-replace-1-1"}, | ||||
| 		{Code: "language-slice-replace-1-2", Name: "language-slice-replace-1-2"}, | ||||
| 	} | ||||
| 	languages2_2 := []*Language{ | ||||
| 		{Code: "language-slice-replace-2-1", Name: "language-slice-replace-2-1"}, | ||||
| 		{Code: "language-slice-replace-2-2", Name: "language-slice-replace-2-2"}, | ||||
| 	} | ||||
| 	languages2_3 := &Language{Code: "language-slice-replace-3", Name: "language-slice-replace-3"} | ||||
| 	DB.Create(&languages2_1) | ||||
| 	DB.Create(&languages2_2) | ||||
| 	DB.Create(&languages2_3) | ||||
| 
 | ||||
| 	// Replace
 | ||||
| 	DB.Model(&users).Association("Languages").Replace(&languages2_1, &languages2_2, languages2_3) | ||||
| 
 | ||||
| 	AssertAssociationCount(t, users, "Languages", 5, "After Replace") | ||||
| 
 | ||||
| 	// Delete
 | ||||
| 	if err := DB.Model(&users).Association("Languages").Delete(&users[2].Languages); err != nil { | ||||
| 		t.Errorf("no error should happend when deleting language, but got %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	AssertAssociationCount(t, users, "Languages", 4, "after delete") | ||||
| 
 | ||||
| 	if err := DB.Model(&users).Association("Languages").Delete(users[0].Languages[0], users[1].Languages[1]); err != nil { | ||||
| 		t.Errorf("no error should happend when deleting language, but got %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	AssertAssociationCount(t, users, "Languages", 2, "after delete") | ||||
| 
 | ||||
| 	// Clear
 | ||||
| 	DB.Model(&users).Association("Languages").Clear() | ||||
| 	AssertAssociationCount(t, users, "Languages", 0, "After Clear") | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Jinzhu
						Jinzhu