add auto fields filtering feature for function Scan of package gdb; mark funtcion Struct/Structs deprecated for gdb.Model

This commit is contained in:
John Guo 2021-06-22 21:48:56 +08:00
parent 69dd5db774
commit d6ea2220f7
5 changed files with 156 additions and 79 deletions

View File

@ -8,6 +8,7 @@ package gdb
import ( import (
"fmt" "fmt"
"github.com/gogf/gf/errors/gerror"
"reflect" "reflect"
"github.com/gogf/gf/container/gset" "github.com/gogf/gf/container/gset"
@ -195,6 +196,15 @@ func (m *Model) Array(fieldsAndWhere ...interface{}) ([]Value, error) {
return all.Array(), nil return all.Array(), nil
} }
// Struct retrieves one record from table and converts it into given struct.
// The parameter `pointer` should be type of *struct/**struct. If type **struct is given,
// it can create the struct internally during converting.
//
// Deprecated, use Scan instead.
func (m *Model) Struct(pointer interface{}, where ...interface{}) error {
return m.doStruct(pointer, where...)
}
// Struct retrieves one record from table and converts it into given struct. // Struct retrieves one record from table and converts it into given struct.
// The parameter `pointer` should be type of *struct/**struct. If type **struct is given, // The parameter `pointer` should be type of *struct/**struct. If type **struct is given,
// it can create the struct internally during converting. // it can create the struct internally during converting.
@ -202,24 +212,38 @@ func (m *Model) Array(fieldsAndWhere ...interface{}) ([]Value, error) {
// The optional parameter `where` is the same as the parameter of Model.Where function, // The optional parameter `where` is the same as the parameter of Model.Where function,
// see Model.Where. // see Model.Where.
// //
// Note that it returns sql.ErrNoRows if there's no record retrieved with the given conditions // Note that it returns sql.ErrNoRows if the given parameter `pointer` pointed to a variable that has
// from table and `pointer` is not nil. // default value and there's no record retrieved with the given conditions from table.
// //
// Eg: // Example:
// user := new(User) // user := new(User)
// err := db.Model("user").Where("id", 1).Struct(user) // err := db.Model("user").Where("id", 1).Scan(user)
// //
// user := (*User)(nil) // user := (*User)(nil)
// err := db.Model("user").Where("id", 1).Struct(&user) // err := db.Model("user").Where("id", 1).Scan(&user)
func (m *Model) Struct(pointer interface{}, where ...interface{}) error { func (m *Model) doStruct(pointer interface{}, where ...interface{}) error {
one, err := m.One(where...) model := m
// Auto selecting fields by struct attributes.
if model.fieldsEx == "" && (model.fields == "" || model.fields == "*") {
model = m.Fields(pointer)
}
one, err := model.One(where...)
if err != nil { if err != nil {
return err return err
} }
if err = one.Struct(pointer); err != nil { if err = one.Struct(pointer); err != nil {
return err return err
} }
return m.doWithScanStruct(pointer) return model.doWithScanStruct(pointer)
}
// Structs retrieves records from table and converts them into given struct slice.
// The parameter `pointer` should be type of *[]struct/*[]*struct. It can create and fill the struct
// slice internally during converting.
//
// Deprecated, use Scan instead.
func (m *Model) Structs(pointer interface{}, where ...interface{}) error {
return m.doStructs(pointer, where...)
} }
// Structs retrieves records from table and converts them into given struct slice. // Structs retrieves records from table and converts them into given struct slice.
@ -229,37 +253,45 @@ func (m *Model) Struct(pointer interface{}, where ...interface{}) error {
// The optional parameter `where` is the same as the parameter of Model.Where function, // The optional parameter `where` is the same as the parameter of Model.Where function,
// see Model.Where. // see Model.Where.
// //
// Note that it returns sql.ErrNoRows if there's no record retrieved with the given conditions // Note that it returns sql.ErrNoRows if the given parameter `pointer` pointed to a variable that has
// from table and `pointer` is not empty. // default value and there's no record retrieved with the given conditions from table.
// //
// Eg: // Example:
// users := ([]User)(nil) // users := ([]User)(nil)
// err := db.Model("user").Structs(&users) // err := db.Model("user").Scan(&users)
// //
// users := ([]*User)(nil) // users := ([]*User)(nil)
// err := db.Model("user").Structs(&users) // err := db.Model("user").Scan(&users)
func (m *Model) Structs(pointer interface{}, where ...interface{}) error { func (m *Model) doStructs(pointer interface{}, where ...interface{}) error {
all, err := m.All(where...) model := m
// Auto selecting fields by struct attributes.
if model.fieldsEx == "" && (model.fields == "" || model.fields == "*") {
model = m.Fields(
reflect.New(
reflect.ValueOf(pointer).Elem().Type().Elem(),
).Interface(),
)
}
all, err := model.All(where...)
if err != nil { if err != nil {
return err return err
} }
if err = all.Structs(pointer); err != nil { if err = all.Structs(pointer); err != nil {
return err return err
} }
return m.doWithScanStructs(pointer) return model.doWithScanStructs(pointer)
} }
// Scan automatically calls Struct or Structs function according to the type of parameter `pointer`. // Scan automatically calls Struct or Structs function according to the type of parameter `pointer`.
// It calls function Struct if `pointer` is type of *struct/**struct. // It calls function doStruct if `pointer` is type of *struct/**struct.
// It calls function Structs if `pointer` is type of *[]struct/*[]*struct. // It calls function doStructs if `pointer` is type of *[]struct/*[]*struct.
// //
// The optional parameter `where` is the same as the parameter of Model.Where function, // The optional parameter `where` is the same as the parameter of Model.Where function, see Model.Where.
// see Model.Where.
// //
// Note that it returns sql.ErrNoRows if there's no record retrieved with the given conditions // Note that it returns sql.ErrNoRows if the given parameter `pointer` pointed to a variable that has
// from table. // default value and there's no record retrieved with the given conditions from table.
// //
// Eg: // Example:
// user := new(User) // user := new(User)
// err := db.Model("user").Where("id", 1).Scan(user) // err := db.Model("user").Where("id", 1).Scan(user)
// //
@ -272,16 +304,35 @@ func (m *Model) Structs(pointer interface{}, where ...interface{}) error {
// users := ([]*User)(nil) // users := ([]*User)(nil)
// err := db.Model("user").Scan(&users) // err := db.Model("user").Scan(&users)
func (m *Model) Scan(pointer interface{}, where ...interface{}) error { func (m *Model) Scan(pointer interface{}, where ...interface{}) error {
var reflectType reflect.Type var (
reflectValue reflect.Value
reflectKind reflect.Kind
)
if v, ok := pointer.(reflect.Value); ok { if v, ok := pointer.(reflect.Value); ok {
reflectType = v.Type() reflectValue = v
} else { } else {
reflectType = reflect.TypeOf(pointer) reflectValue = reflect.ValueOf(pointer)
} }
if gstr.Contains(reflectType.String(), "[]") {
return m.Structs(pointer, where...) reflectKind = reflectValue.Kind()
if reflectKind != reflect.Ptr {
return gerror.New(`the parameter "pointer" for function Scan should type of pointer`)
}
for reflectKind == reflect.Ptr {
reflectValue = reflectValue.Elem()
reflectKind = reflectValue.Kind()
}
switch reflectKind {
case reflect.Slice, reflect.Array:
return m.doStructs(pointer, where...)
case reflect.Struct, reflect.Invalid:
return m.doStruct(pointer, where...)
default:
return gerror.New(`element of parameter "pointer" for function Scan should type of struct/*struct/[]struct/[]*struct`)
} }
return m.Struct(pointer, where...)
} }
// ScanList converts `r` to struct slice which contains other complex struct attributes. // ScanList converts `r` to struct slice which contains other complex struct attributes.

View File

@ -153,7 +153,7 @@ func (r Result) MapKeyUint(key string) map[uint]Map {
return m return m
} }
// RecordKeyInt converts `r` to a map[int]Record of which key is specified by `key`. // RecordKeyStr converts `r` to a map[string]Record of which key is specified by `key`.
func (r Result) RecordKeyStr(key string) map[string]Record { func (r Result) RecordKeyStr(key string) map[string]Record {
m := make(map[string]Record) m := make(map[string]Record)
for _, item := range r { for _, item := range r {

View File

@ -899,7 +899,7 @@ func Test_Model_Struct(t *testing.T) {
CreateTime gtime.Time CreateTime gtime.Time
} }
user := new(User) user := new(User)
err := db.Model(table).Where("id=1").Struct(user) err := db.Model(table).Where("id=1").Scan(user)
t.AssertNil(err) t.AssertNil(err)
t.Assert(user.NickName, "name_1") t.Assert(user.NickName, "name_1")
t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00") t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00")
@ -913,7 +913,7 @@ func Test_Model_Struct(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
user := new(User) user := new(User)
err := db.Model(table).Where("id=1").Struct(user) err := db.Model(table).Where("id=1").Scan(user)
t.AssertNil(err) t.AssertNil(err)
t.Assert(user.NickName, "name_1") t.Assert(user.NickName, "name_1")
t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00") t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00")
@ -928,7 +928,7 @@ func Test_Model_Struct(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
user := (*User)(nil) user := (*User)(nil)
err := db.Model(table).Where("id=1").Struct(&user) err := db.Model(table).Where("id=1").Scan(&user)
t.AssertNil(err) t.AssertNil(err)
t.Assert(user.NickName, "name_1") t.Assert(user.NickName, "name_1")
t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00") t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00")
@ -960,7 +960,7 @@ func Test_Model_Struct(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
user := new(User) user := new(User)
err := db.Model(table).Where("id=-1").Struct(user) err := db.Model(table).Where("id=-1").Scan(user)
t.Assert(err, sql.ErrNoRows) t.Assert(err, sql.ErrNoRows)
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
@ -972,7 +972,7 @@ func Test_Model_Struct(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
var user *User var user *User
err := db.Model(table).Where("id=-1").Struct(&user) err := db.Model(table).Where("id=-1").Scan(&user)
t.AssertNil(err) t.AssertNil(err)
}) })
} }
@ -992,7 +992,7 @@ func Test_Model_Struct_CustomType(t *testing.T) {
CreateTime gtime.Time CreateTime gtime.Time
} }
user := new(User) user := new(User)
err := db.Model(table).Where("id=1").Struct(user) err := db.Model(table).Where("id=1").Scan(user)
t.AssertNil(err) t.AssertNil(err)
t.Assert(user.NickName, "name_1") t.Assert(user.NickName, "name_1")
t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00") t.Assert(user.CreateTime.String(), "2018-10-24 10:00:00")
@ -1012,7 +1012,7 @@ func Test_Model_Structs(t *testing.T) {
CreateTime gtime.Time CreateTime gtime.Time
} }
var users []User var users []User
err := db.Model(table).Order("id asc").Structs(&users) err := db.Model(table).Order("id asc").Scan(&users)
if err != nil { if err != nil {
gtest.Error(err) gtest.Error(err)
} }
@ -1035,7 +1035,7 @@ func Test_Model_Structs(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
var users []*User var users []*User
err := db.Model(table).Order("id asc").Structs(&users) err := db.Model(table).Order("id asc").Scan(&users)
if err != nil { if err != nil {
gtest.Error(err) gtest.Error(err)
} }
@ -1081,38 +1081,40 @@ func Test_Model_Structs(t *testing.T) {
CreateTime *gtime.Time CreateTime *gtime.Time
} }
var users []*User var users []*User
err := db.Model(table).Where("id<0").Structs(&users) err := db.Model(table).Where("id<0").Scan(&users)
t.AssertNil(err) t.AssertNil(err)
}) })
} }
func Test_Model_StructsWithJsonTag(t *testing.T) { // JSON tag is only used for JSON Marshal/Unmarshal, DO NOT use it in multiple purposes!
table := createInitTable() //func Test_Model_StructsWithJsonTag(t *testing.T) {
defer dropTable(table) // table := createInitTable()
// defer dropTable(table)
gtest.C(t, func(t *gtest.T) { //
type User struct { // db.SetDebug(true)
Uid int `json:"id"` // gtest.C(t, func(t *gtest.T) {
Passport string // type User struct {
Password string // Uid int `json:"id"`
Name string `json:"nick_name"` // Passport string
Time gtime.Time `json:"create_time"` // Password string
} // Name string `json:"nick_name"`
var users []User // Time gtime.Time `json:"create_time"`
err := db.Model(table).Order("id asc").Structs(&users) // }
if err != nil { // var users []User
gtest.Error(err) // err := db.Model(table).Order("id asc").Scan(&users)
} // if err != nil {
t.Assert(len(users), TableSize) // gtest.Error(err)
t.Assert(users[0].Uid, 1) // }
t.Assert(users[1].Uid, 2) // t.Assert(len(users), TableSize)
t.Assert(users[2].Uid, 3) // t.Assert(users[0].Uid, 1)
t.Assert(users[0].Name, "name_1") // t.Assert(users[1].Uid, 2)
t.Assert(users[1].Name, "name_2") // t.Assert(users[2].Uid, 3)
t.Assert(users[2].Name, "name_3") // t.Assert(users[0].Name, "name_1")
t.Assert(users[0].Time.String(), "2018-10-24 10:00:00") // t.Assert(users[1].Name, "name_2")
}) // t.Assert(users[2].Name, "name_3")
} // t.Assert(users[0].Time.String(), "2018-10-24 10:00:00")
// })
//}
func Test_Model_Scan(t *testing.T) { func Test_Model_Scan(t *testing.T) {
table := createInitTable() table := createInitTable()
@ -3098,7 +3100,7 @@ func Test_TimeZoneInsert(t *testing.T) {
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
_, _ = db.Model(tableName).Unscoped().Insert(u) _, _ = db.Model(tableName).Unscoped().Insert(u)
userEntity := &User{} userEntity := &User{}
err := db.Model(tableName).Where("id", 1).Unscoped().Struct(&userEntity) err := db.Model(tableName).Where("id", 1).Unscoped().Scan(&userEntity)
t.AssertNil(err) t.AssertNil(err)
t.Assert(userEntity.CreatedAt.String(), "2020-11-22 04:23:45") t.Assert(userEntity.CreatedAt.String(), "2020-11-22 04:23:45")
t.Assert(userEntity.UpdatedAt.String(), "2020-11-22 05:23:45") t.Assert(userEntity.UpdatedAt.String(), "2020-11-22 05:23:45")
@ -3129,7 +3131,7 @@ func Test_Model_Fields_Map_Struct(t *testing.T) {
XXX_TYPE int XXX_TYPE int
} }
var a = A{} var a = A{}
err := db.Model(table).Fields(a).Where("id", 1).Struct(&a) err := db.Model(table).Fields(a).Where("id", 1).Scan(&a)
t.AssertNil(err) t.AssertNil(err)
t.Assert(a.ID, 1) t.Assert(a.ID, 1)
t.Assert(a.PASSPORT, "user_1") t.Assert(a.PASSPORT, "user_1")
@ -3143,7 +3145,7 @@ func Test_Model_Fields_Map_Struct(t *testing.T) {
XXX_TYPE int XXX_TYPE int
} }
var a *A var a *A
err := db.Model(table).Fields(a).Where("id", 1).Struct(&a) err := db.Model(table).Fields(a).Where("id", 1).Scan(&a)
t.AssertNil(err) t.AssertNil(err)
t.Assert(a.ID, 1) t.Assert(a.ID, 1)
t.Assert(a.PASSPORT, "user_1") t.Assert(a.PASSPORT, "user_1")
@ -3157,7 +3159,7 @@ func Test_Model_Fields_Map_Struct(t *testing.T) {
XXX_TYPE int XXX_TYPE int
} }
var a *A var a *A
err := db.Model(table).Fields(&a).Where("id", 1).Struct(&a) err := db.Model(table).Fields(&a).Where("id", 1).Scan(&a)
t.AssertNil(err) t.AssertNil(err)
t.Assert(a.ID, 1) t.Assert(a.ID, 1)
t.Assert(a.PASSPORT, "user_1") t.Assert(a.PASSPORT, "user_1")

View File

@ -121,7 +121,7 @@ func Test_Struct_Pointer_Attribute(t *testing.T) {
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
user := new(User) user := new(User)
err := db.Model(table).Struct(user, "id=1") err := db.Model(table).Scan(user, "id=1")
t.AssertNil(err) t.AssertNil(err)
t.Assert(*user.Id, 1) t.Assert(*user.Id, 1)
t.Assert(*user.Passport, "user_1") t.Assert(*user.Passport, "user_1")
@ -130,7 +130,7 @@ func Test_Struct_Pointer_Attribute(t *testing.T) {
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
var user *User var user *User
err := db.Model(table).Struct(&user, "id=1") err := db.Model(table).Scan(&user, "id=1")
t.AssertNil(err) t.AssertNil(err)
t.Assert(*user.Id, 1) t.Assert(*user.Id, 1)
t.Assert(*user.Passport, "user_1") t.Assert(*user.Passport, "user_1")
@ -201,7 +201,7 @@ func Test_Structs_Pointer_Attribute(t *testing.T) {
// Structs // Structs
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
users := make([]User, 0) users := make([]User, 0)
err := db.Model(table).Structs(&users, "id < 3") err := db.Model(table).Scan(&users, "id < 3")
t.AssertNil(err) t.AssertNil(err)
t.Assert(len(users), 2) t.Assert(len(users), 2)
t.Assert(*users[0].Id, 1) t.Assert(*users[0].Id, 1)
@ -211,7 +211,7 @@ func Test_Structs_Pointer_Attribute(t *testing.T) {
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
users := make([]*User, 0) users := make([]*User, 0)
err := db.Model(table).Structs(&users, "id < 3") err := db.Model(table).Scan(&users, "id < 3")
t.AssertNil(err) t.AssertNil(err)
t.Assert(len(users), 2) t.Assert(len(users), 2)
t.Assert(*users[0].Id, 1) t.Assert(*users[0].Id, 1)
@ -221,7 +221,7 @@ func Test_Structs_Pointer_Attribute(t *testing.T) {
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
var users []User var users []User
err := db.Model(table).Structs(&users, "id < 3") err := db.Model(table).Scan(&users, "id < 3")
t.AssertNil(err) t.AssertNil(err)
t.Assert(len(users), 2) t.Assert(len(users), 2)
t.Assert(*users[0].Id, 1) t.Assert(*users[0].Id, 1)
@ -231,7 +231,7 @@ func Test_Structs_Pointer_Attribute(t *testing.T) {
}) })
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
var users []*User var users []*User
err := db.Model(table).Structs(&users, "id < 3") err := db.Model(table).Scan(&users, "id < 3")
t.AssertNil(err) t.AssertNil(err)
t.Assert(len(users), 2) t.Assert(len(users), 2)
t.Assert(*users[0].Id, 1) t.Assert(*users[0].Id, 1)
@ -254,7 +254,7 @@ func Test_Struct_Empty(t *testing.T) {
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
user := new(User) user := new(User)
err := db.Model(table).Where("id=100").Struct(user) err := db.Model(table).Where("id=100").Scan(user)
t.Assert(err, sql.ErrNoRows) t.Assert(err, sql.ErrNoRows)
t.AssertNE(user, nil) t.AssertNE(user, nil)
}) })
@ -269,7 +269,7 @@ func Test_Struct_Empty(t *testing.T) {
gtest.C(t, func(t *gtest.T) { gtest.C(t, func(t *gtest.T) {
var user *User var user *User
err := db.Model(table).Where("id=100").Struct(&user) err := db.Model(table).Where("id=100").Scan(&user)
t.AssertNil(err) t.AssertNil(err)
t.Assert(user, nil) t.Assert(user, nil)
}) })
@ -452,3 +452,27 @@ func Test_Model_Scan_Map(t *testing.T) {
t.Assert(users[9].CreateTime.String(), CreateTime) t.Assert(users[9].CreateTime.String(), CreateTime)
}) })
} }
func Test_Scan_AutoFilteringByStructAttributes(t *testing.T) {
table := createInitTable()
defer dropTable(table)
type User struct {
Id int
Passport string
}
//db.SetDebug(true)
gtest.C(t, func(t *gtest.T) {
var user *User
err := db.Model(table).OrderAsc("id").Scan(&user)
t.AssertNil(err)
t.Assert(user.Id, 1)
})
gtest.C(t, func(t *gtest.T) {
var users []User
err := db.Model(table).OrderAsc("id").Scan(&users)
t.AssertNil(err)
t.Assert(len(users), TableSize)
t.Assert(users[0].Id, 1)
})
}

View File

@ -87,7 +87,7 @@ func Test_Types(t *testing.T) {
TinyInt bool TinyInt bool
} }
var obj *T var obj *T
err = db.Model("types").Struct(&obj) err = db.Model("types").Scan(&obj)
t.AssertNil(err) t.AssertNil(err)
t.Assert(obj.Id, 1) t.Assert(obj.Id, 1)
t.Assert(obj.Blob, data["blob"]) t.Assert(obj.Blob, data["blob"])