Add test case for embedded value selects (#5901)
* Add test case for embedded value selects * Revert recycle struct optimisation to avoid pointer overwrites
This commit is contained in:
		
							parent
							
								
									d9525d4da4
								
							
						
					
					
						commit
						4ec73c9bf4
					
				
							
								
								
									
										12
									
								
								scan.go
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								scan.go
									
									
									
									
									
								
							| @ -65,7 +65,6 @@ func (db *DB) scanIntoStruct(rows Rows, reflectValue reflect.Value, values []int | |||||||
| 
 | 
 | ||||||
| 	db.RowsAffected++ | 	db.RowsAffected++ | ||||||
| 	db.AddError(rows.Scan(values...)) | 	db.AddError(rows.Scan(values...)) | ||||||
| 
 |  | ||||||
| 	joinedSchemaMap := make(map[*schema.Field]interface{}) | 	joinedSchemaMap := make(map[*schema.Field]interface{}) | ||||||
| 	for idx, field := range fields { | 	for idx, field := range fields { | ||||||
| 		if field == nil { | 		if field == nil { | ||||||
| @ -241,9 +240,8 @@ func Scan(rows Rows, db *DB, mode ScanMode) { | |||||||
| 		switch reflectValue.Kind() { | 		switch reflectValue.Kind() { | ||||||
| 		case reflect.Slice, reflect.Array: | 		case reflect.Slice, reflect.Array: | ||||||
| 			var ( | 			var ( | ||||||
| 				elem             reflect.Value | 				elem        reflect.Value | ||||||
| 				recyclableStruct = reflect.New(reflectValueType) | 				isArrayKind = reflectValue.Kind() == reflect.Array | ||||||
| 				isArrayKind      = reflectValue.Kind() == reflect.Array |  | ||||||
| 			) | 			) | ||||||
| 
 | 
 | ||||||
| 			if !update || reflectValue.Len() == 0 { | 			if !update || reflectValue.Len() == 0 { | ||||||
| @ -275,11 +273,7 @@ func Scan(rows Rows, db *DB, mode ScanMode) { | |||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 				} else { | 				} else { | ||||||
| 					if isPtr && db.RowsAffected > 0 { | 					elem = reflect.New(reflectValueType) | ||||||
| 						elem = reflect.New(reflectValueType) |  | ||||||
| 					} else { |  | ||||||
| 						elem = recyclableStruct |  | ||||||
| 					} |  | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				db.scanIntoStruct(rows, elem, values, fields, joinFields) | 				db.scanIntoStruct(rows, elem, values, fields, joinFields) | ||||||
|  | |||||||
| @ -36,7 +36,7 @@ func TestEmbeddedStruct(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	type EngadgetPost struct { | 	type EngadgetPost struct { | ||||||
| 		BasePost BasePost `gorm:"Embedded"` | 		BasePost BasePost `gorm:"Embedded"` | ||||||
| 		Author   Author   `gorm:"Embedded;EmbeddedPrefix:author_"` // Embedded struct
 | 		Author   *Author  `gorm:"Embedded;EmbeddedPrefix:author_"` // Embedded struct
 | ||||||
| 		ImageUrl string | 		ImageUrl string | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -74,13 +74,27 @@ func TestEmbeddedStruct(t *testing.T) { | |||||||
| 		t.Errorf("embedded struct's value should be scanned correctly") | 		t.Errorf("embedded struct's value should be scanned correctly") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	DB.Save(&EngadgetPost{BasePost: BasePost{Title: "engadget_news"}}) | 	DB.Save(&EngadgetPost{BasePost: BasePost{Title: "engadget_news"}, Author: &Author{Name: "Edward"}}) | ||||||
|  | 	DB.Save(&EngadgetPost{BasePost: BasePost{Title: "engadget_article"}, Author: &Author{Name: "George"}}) | ||||||
| 	var egNews EngadgetPost | 	var egNews EngadgetPost | ||||||
| 	if err := DB.First(&egNews, "title = ?", "engadget_news").Error; err != nil { | 	if err := DB.First(&egNews, "title = ?", "engadget_news").Error; err != nil { | ||||||
| 		t.Errorf("no error should happen when query with embedded struct, but got %v", err) | 		t.Errorf("no error should happen when query with embedded struct, but got %v", err) | ||||||
| 	} else if egNews.BasePost.Title != "engadget_news" { | 	} else if egNews.BasePost.Title != "engadget_news" { | ||||||
| 		t.Errorf("embedded struct's value should be scanned correctly") | 		t.Errorf("embedded struct's value should be scanned correctly") | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	var egPosts []EngadgetPost | ||||||
|  | 	if err := DB.Order("author_name asc").Find(&egPosts).Error; err != nil { | ||||||
|  | 		t.Fatalf("no error should happen when query with embedded struct, but got %v", err) | ||||||
|  | 	} | ||||||
|  | 	expectAuthors := []string{"Edward", "George"} | ||||||
|  | 	for i, post := range egPosts { | ||||||
|  | 		t.Log(i, post.Author) | ||||||
|  | 		if want := expectAuthors[i]; post.Author.Name != want { | ||||||
|  | 			t.Errorf("expected author %s got %s", want, post.Author.Name) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestEmbeddedPointerTypeStruct(t *testing.T) { | func TestEmbeddedPointerTypeStruct(t *testing.T) { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Edward McFarlane
						Edward McFarlane