mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 11:18:50 +08:00
Merge 15bbcc8f539ce3b61616768ad15c2c7e55452654 into 1534abdb050acaa5aa6ef27f94e91bca97e6faa3
This commit is contained in:
commit
c4bc6cf39b
@ -726,3 +726,33 @@ func Test_Issue4093(t *testing.T) {
|
|||||||
t.Assert(client.PostContent(ctx, "/test"), `{"page":1,"pageSize":10,"pagination":true,"name":"john","number":1}`)
|
t.Assert(client.PostContent(ctx, "/test"), `{"page":1,"pageSize":10,"pagination":true,"name":"john","number":1}`)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://github.com/gogf/gf/issues/4193
|
||||||
|
func Test_Issue4193(t *testing.T) {
|
||||||
|
type Req struct {
|
||||||
|
g.Meta `method:"post" mime:"multipart/form-data"`
|
||||||
|
File *ghttp.UploadFile `v:"required" type:"file"` // File is required
|
||||||
|
}
|
||||||
|
type Res struct{}
|
||||||
|
|
||||||
|
s := g.Server(guid.S())
|
||||||
|
s.BindMiddlewareDefault(ghttp.MiddlewareHandlerResponse)
|
||||||
|
s.BindHandler("/upload", func(ctx context.Context, req *Req) (res *Res, err error) {
|
||||||
|
return
|
||||||
|
})
|
||||||
|
s.SetDumpRouterMap(false)
|
||||||
|
s.SetAccessLogEnabled(false)
|
||||||
|
s.SetErrorLogEnabled(false)
|
||||||
|
s.Start()
|
||||||
|
defer s.Shutdown()
|
||||||
|
time.Sleep(100 * time.Millisecond)
|
||||||
|
|
||||||
|
gtest.C(t, func(t *gtest.T) {
|
||||||
|
client := g.Client()
|
||||||
|
client.SetPrefix(fmt.Sprintf("http://127.0.0.1:%d", s.GetListenedPort()))
|
||||||
|
content := client.PostContent(ctx, "/upload", g.Map{
|
||||||
|
"file": "",
|
||||||
|
})
|
||||||
|
t.Assert(content, `{"code":51,"message":"The File field is required","data":null}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
@ -18,7 +18,7 @@ package gconv
|
|||||||
// TODO: change `paramKeyToAttrMap` to `ScanOption` to be more scalable; add `DeepCopy` option for `ScanOption`.
|
// TODO: change `paramKeyToAttrMap` to `ScanOption` to be more scalable; add `DeepCopy` option for `ScanOption`.
|
||||||
func Scan(srcValue any, dstPointer any, paramKeyToAttrMap ...map[string]string) (err error) {
|
func Scan(srcValue any, dstPointer any, paramKeyToAttrMap ...map[string]string) (err error) {
|
||||||
option := ScanOption{
|
option := ScanOption{
|
||||||
ContinueOnError: true,
|
ContinueOnError: false,
|
||||||
}
|
}
|
||||||
if len(paramKeyToAttrMap) > 0 {
|
if len(paramKeyToAttrMap) > 0 {
|
||||||
option.ParamKeyToAttrMap = paramKeyToAttrMap[0]
|
option.ParamKeyToAttrMap = paramKeyToAttrMap[0]
|
||||||
|
@ -395,7 +395,11 @@ func (c *Converter) doConvertForDefault(in doConvertInput, option ConvertOption)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// custom converter.
|
// custom converter.
|
||||||
dstReflectValue, ok, err := c.callCustomConverterWithRefer(fromReflectValue, referReflectValue)
|
var (
|
||||||
|
ok bool
|
||||||
|
dstReflectValue reflect.Value
|
||||||
|
)
|
||||||
|
dstReflectValue, ok, err = c.callCustomConverterWithRefer(fromReflectValue, referReflectValue)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -415,7 +419,7 @@ func (c *Converter) doConvertForDefault(in doConvertInput, option ConvertOption)
|
|||||||
switch referReflectValue.Kind() {
|
switch referReflectValue.Kind() {
|
||||||
case reflect.Ptr:
|
case reflect.Ptr:
|
||||||
// Type converting for custom type pointers.
|
// Type converting for custom type pointers.
|
||||||
// Eg:
|
// Example:
|
||||||
// type PayMode int
|
// type PayMode int
|
||||||
// type Req struct{
|
// type Req struct{
|
||||||
// Mode *PayMode
|
// Mode *PayMode
|
||||||
|
@ -45,7 +45,11 @@ func (c *Converter) Float32(any any) (float32, error) {
|
|||||||
}
|
}
|
||||||
return 0, nil
|
return 0, nil
|
||||||
case reflect.String:
|
case reflect.String:
|
||||||
f, err := strconv.ParseFloat(rv.String(), 32)
|
s := rv.String()
|
||||||
|
if s == "" {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
f, err := strconv.ParseFloat(s, 32)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, gerror.WrapCodef(
|
return 0, gerror.WrapCodef(
|
||||||
gcode.CodeInvalidParameter, err, "converting string to float32 failed for: %v", any,
|
gcode.CodeInvalidParameter, err, "converting string to float32 failed for: %v", any,
|
||||||
@ -68,6 +72,9 @@ func (c *Converter) Float32(any any) (float32, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
if s == "" {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
v, err := strconv.ParseFloat(s, 32)
|
v, err := strconv.ParseFloat(s, 32)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, gerror.WrapCodef(
|
return 0, gerror.WrapCodef(
|
||||||
@ -112,7 +119,11 @@ func (c *Converter) Float64(any any) (float64, error) {
|
|||||||
}
|
}
|
||||||
return 0, nil
|
return 0, nil
|
||||||
case reflect.String:
|
case reflect.String:
|
||||||
f, err := strconv.ParseFloat(rv.String(), 64)
|
s := rv.String()
|
||||||
|
if s == "" {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
f, err := strconv.ParseFloat(s, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, gerror.WrapCodef(
|
return 0, gerror.WrapCodef(
|
||||||
gcode.CodeInvalidParameter, err, "converting string to float64 failed for: %v", any,
|
gcode.CodeInvalidParameter, err, "converting string to float64 failed for: %v", any,
|
||||||
@ -135,6 +146,9 @@ func (c *Converter) Float64(any any) (float64, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
if s == "" {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
v, err := strconv.ParseFloat(s, 64)
|
v, err := strconv.ParseFloat(s, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, gerror.WrapCodef(
|
return 0, gerror.WrapCodef(
|
||||||
|
@ -88,6 +88,9 @@ func (c *Converter) doMapConvert(
|
|||||||
value any, recursive RecursiveType, mustMapReturn bool, option MapOption,
|
value any, recursive RecursiveType, mustMapReturn bool, option MapOption,
|
||||||
) (map[string]any, error) {
|
) (map[string]any, error) {
|
||||||
if value == nil {
|
if value == nil {
|
||||||
|
if mustMapReturn {
|
||||||
|
return map[string]any{}, nil
|
||||||
|
}
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
// It redirects to its underlying value if it has implemented interface iVal.
|
// It redirects to its underlying value if it has implemented interface iVal.
|
||||||
@ -119,6 +122,10 @@ func (c *Converter) doMapConvert(
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
if len(r) == 0 && mustMapReturn {
|
||||||
|
return map[string]any{}, nil
|
||||||
|
}
|
||||||
|
// if r is not empty, which means it fails converting to map.
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
case []byte:
|
case []byte:
|
||||||
@ -128,6 +135,10 @@ func (c *Converter) doMapConvert(
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
if len(r) == 0 && mustMapReturn {
|
||||||
|
return map[string]any{}, nil
|
||||||
|
}
|
||||||
|
// if r is not empty, which means it fails converting to map.
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
case map[interface{}]interface{}:
|
case map[interface{}]interface{}:
|
||||||
@ -328,6 +339,7 @@ func (c *Converter) doMapConvert(
|
|||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
return nil, nil
|
return nil, nil
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
@ -160,10 +160,11 @@ func (c *Converter) Struct(params, pointer any, option ...StructOption) (err err
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if paramsMap == nil {
|
if paramsMap == nil {
|
||||||
|
// fails converting params to map, it so cannot be converted to struct pointer.
|
||||||
return gerror.NewCodef(
|
return gerror.NewCodef(
|
||||||
gcode.CodeInvalidParameter,
|
gcode.CodeInvalidParameter,
|
||||||
`convert params from "%#v" to "map[string]any" failed`,
|
`convert params "%v" to "%s" failed`,
|
||||||
params,
|
params, pointerReflectValue.Type(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -505,8 +506,7 @@ func (c *Converter) bindVarToReflectValue(structFieldValue reflect.Value, value
|
|||||||
case reflect.Struct:
|
case reflect.Struct:
|
||||||
// Recursively converting for struct attribute.
|
// Recursively converting for struct attribute.
|
||||||
if err = c.Struct(value, structFieldValue, option); err != nil {
|
if err = c.Struct(value, structFieldValue, option); err != nil {
|
||||||
// Note there's reflect conversion mechanism here.
|
return err
|
||||||
structFieldValue.Set(reflect.ValueOf(value).Convert(structFieldValue.Type()))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note that the slice element might be type of struct,
|
// Note that the slice element might be type of struct,
|
||||||
@ -637,6 +637,8 @@ func (c *Converter) bindVarToReflectValue(structFieldValue reflect.Value, value
|
|||||||
elem := item.Elem()
|
elem := item.Elem()
|
||||||
if err = c.bindVarToReflectValue(elem, value, option); err == nil {
|
if err = c.bindVarToReflectValue(elem, value, option); err == nil {
|
||||||
structFieldValue.Set(elem.Addr())
|
structFieldValue.Set(elem.Addr())
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Not empty pointer, it assigns values to it.
|
// Not empty pointer, it assigns values to it.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user