mirror of
https://github.com/labstack/echo.git
synced 2025-01-20 02:59:54 +02:00
Default binder can use UnmarshalParams(params []string) error
interface to bind multiple input values at one go. (#2607)
This commit is contained in:
parent
a3b0ba24d3
commit
c57fcb3746
64
bind.go
64
bind.go
@ -30,6 +30,13 @@ type BindUnmarshaler interface {
|
|||||||
UnmarshalParam(param string) error
|
UnmarshalParam(param string) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// bindMultipleUnmarshaler is used by binder to unmarshal multiple values from request at once to
|
||||||
|
// type implementing this interface. For example request could have multiple query fields `?a=1&a=2&b=test` in that case
|
||||||
|
// for `a` following slice `["1", "2"] will be passed to unmarshaller.
|
||||||
|
type bindMultipleUnmarshaler interface {
|
||||||
|
UnmarshalParams(params []string) error
|
||||||
|
}
|
||||||
|
|
||||||
// BindPathParams binds path params to bindable object
|
// BindPathParams binds path params to bindable object
|
||||||
func (b *DefaultBinder) BindPathParams(c Context, i interface{}) error {
|
func (b *DefaultBinder) BindPathParams(c Context, i interface{}) error {
|
||||||
names := c.ParamNames()
|
names := c.ParamNames()
|
||||||
@ -217,8 +224,15 @@ func (b *DefaultBinder) bindData(destination interface{}, data map[string][]stri
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ok, err := unmarshalInputsToField(typeField.Type.Kind(), inputValue, structField); ok {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
// Call this first, in case we're dealing with an alias to an array type
|
// Call this first, in case we're dealing with an alias to an array type
|
||||||
if ok, err := unmarshalField(typeField.Type.Kind(), inputValue[0], structField); ok {
|
if ok, err := unmarshalInputToField(typeField.Type.Kind(), inputValue[0], structField); ok {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -245,7 +259,7 @@ func (b *DefaultBinder) bindData(destination interface{}, data map[string][]stri
|
|||||||
|
|
||||||
func setWithProperType(valueKind reflect.Kind, val string, structField reflect.Value) error {
|
func setWithProperType(valueKind reflect.Kind, val string, structField reflect.Value) error {
|
||||||
// But also call it here, in case we're dealing with an array of BindUnmarshalers
|
// But also call it here, in case we're dealing with an array of BindUnmarshalers
|
||||||
if ok, err := unmarshalField(valueKind, val, structField); ok {
|
if ok, err := unmarshalInputToField(valueKind, val, structField); ok {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -286,35 +300,41 @@ func setWithProperType(valueKind reflect.Kind, val string, structField reflect.V
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func unmarshalField(valueKind reflect.Kind, val string, field reflect.Value) (bool, error) {
|
func unmarshalInputsToField(valueKind reflect.Kind, values []string, field reflect.Value) (bool, error) {
|
||||||
switch valueKind {
|
if valueKind == reflect.Ptr {
|
||||||
case reflect.Ptr:
|
if field.IsNil() {
|
||||||
return unmarshalFieldPtr(val, field)
|
field.Set(reflect.New(field.Type().Elem()))
|
||||||
default:
|
|
||||||
return unmarshalFieldNonPtr(val, field)
|
|
||||||
}
|
}
|
||||||
|
field = field.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
fieldIValue := field.Addr().Interface()
|
||||||
|
unmarshaler, ok := fieldIValue.(bindMultipleUnmarshaler)
|
||||||
|
if !ok {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
return true, unmarshaler.UnmarshalParams(values)
|
||||||
}
|
}
|
||||||
|
|
||||||
func unmarshalFieldNonPtr(value string, field reflect.Value) (bool, error) {
|
func unmarshalInputToField(valueKind reflect.Kind, val string, field reflect.Value) (bool, error) {
|
||||||
fieldIValue := field.Addr().Interface()
|
if valueKind == reflect.Ptr {
|
||||||
if unmarshaler, ok := fieldIValue.(BindUnmarshaler); ok {
|
if field.IsNil() {
|
||||||
return true, unmarshaler.UnmarshalParam(value)
|
field.Set(reflect.New(field.Type().Elem()))
|
||||||
}
|
}
|
||||||
if unmarshaler, ok := fieldIValue.(encoding.TextUnmarshaler); ok {
|
field = field.Elem()
|
||||||
return true, unmarshaler.UnmarshalText([]byte(value))
|
}
|
||||||
|
|
||||||
|
fieldIValue := field.Addr().Interface()
|
||||||
|
switch unmarshaler := fieldIValue.(type) {
|
||||||
|
case BindUnmarshaler:
|
||||||
|
return true, unmarshaler.UnmarshalParam(val)
|
||||||
|
case encoding.TextUnmarshaler:
|
||||||
|
return true, unmarshaler.UnmarshalText([]byte(val))
|
||||||
}
|
}
|
||||||
|
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func unmarshalFieldPtr(value string, field reflect.Value) (bool, error) {
|
|
||||||
if field.IsNil() {
|
|
||||||
// Initialize the pointer to a nil value
|
|
||||||
field.Set(reflect.New(field.Type().Elem()))
|
|
||||||
}
|
|
||||||
return unmarshalFieldNonPtr(value, field.Elem())
|
|
||||||
}
|
|
||||||
|
|
||||||
func setIntField(value string, bitSize int, field reflect.Value) error {
|
func setIntField(value string, bitSize int, field reflect.Value) error {
|
||||||
if value == "" {
|
if value == "" {
|
||||||
value = "0"
|
value = "0"
|
||||||
|
302
bind_test.go
302
bind_test.go
@ -8,6 +8,7 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"mime/multipart"
|
"mime/multipart"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -653,49 +654,6 @@ func TestBindSetWithProperType(t *testing.T) {
|
|||||||
assert.Error(t, setWithProperType(typ.Field(0).Type.Kind(), "5", val.Field(0)))
|
assert.Error(t, setWithProperType(typ.Field(0).Type.Kind(), "5", val.Field(0)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBindSetFields(t *testing.T) {
|
|
||||||
|
|
||||||
ts := new(bindTestStruct)
|
|
||||||
val := reflect.ValueOf(ts).Elem()
|
|
||||||
// Int
|
|
||||||
if assert.NoError(t, setIntField("5", 0, val.FieldByName("I"))) {
|
|
||||||
assert.Equal(t, 5, ts.I)
|
|
||||||
}
|
|
||||||
if assert.NoError(t, setIntField("", 0, val.FieldByName("I"))) {
|
|
||||||
assert.Equal(t, 0, ts.I)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uint
|
|
||||||
if assert.NoError(t, setUintField("10", 0, val.FieldByName("UI"))) {
|
|
||||||
assert.Equal(t, uint(10), ts.UI)
|
|
||||||
}
|
|
||||||
if assert.NoError(t, setUintField("", 0, val.FieldByName("UI"))) {
|
|
||||||
assert.Equal(t, uint(0), ts.UI)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Float
|
|
||||||
if assert.NoError(t, setFloatField("15.5", 0, val.FieldByName("F32"))) {
|
|
||||||
assert.Equal(t, float32(15.5), ts.F32)
|
|
||||||
}
|
|
||||||
if assert.NoError(t, setFloatField("", 0, val.FieldByName("F32"))) {
|
|
||||||
assert.Equal(t, float32(0.0), ts.F32)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bool
|
|
||||||
if assert.NoError(t, setBoolField("true", val.FieldByName("B"))) {
|
|
||||||
assert.Equal(t, true, ts.B)
|
|
||||||
}
|
|
||||||
if assert.NoError(t, setBoolField("", val.FieldByName("B"))) {
|
|
||||||
assert.Equal(t, false, ts.B)
|
|
||||||
}
|
|
||||||
|
|
||||||
ok, err := unmarshalFieldNonPtr("2016-12-06T19:09:05Z", val.FieldByName("T"))
|
|
||||||
if assert.NoError(t, err) {
|
|
||||||
assert.Equal(t, ok, true)
|
|
||||||
assert.Equal(t, Timestamp(time.Date(2016, 12, 6, 19, 9, 5, 0, time.UTC)), ts.T)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func BenchmarkBindbindDataWithTags(b *testing.B) {
|
func BenchmarkBindbindDataWithTags(b *testing.B) {
|
||||||
b.ReportAllocs()
|
b.ReportAllocs()
|
||||||
ts := new(bindTestStructWithTags)
|
ts := new(bindTestStructWithTags)
|
||||||
@ -1138,3 +1096,261 @@ func TestDefaultBinder_BindBody(t *testing.T) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type unixTimestamp struct {
|
||||||
|
Time time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *unixTimestamp) UnmarshalParam(param string) error {
|
||||||
|
n, err := strconv.ParseInt(param, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("'%s' is not an integer", param)
|
||||||
|
}
|
||||||
|
*t = unixTimestamp{Time: time.Unix(n, 0)}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
type IntArrayA []int
|
||||||
|
|
||||||
|
// UnmarshalParam converts value to *Int64Slice. This allows the API to accept
|
||||||
|
// a comma-separated list of integers as a query parameter.
|
||||||
|
func (i *IntArrayA) UnmarshalParam(value string) error {
|
||||||
|
var values = strings.Split(value, ",")
|
||||||
|
var numbers = make([]int, 0, len(values))
|
||||||
|
|
||||||
|
for _, v := range values {
|
||||||
|
n, err := strconv.ParseInt(v, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("'%s' is not an integer", v)
|
||||||
|
}
|
||||||
|
|
||||||
|
numbers = append(numbers, int(n))
|
||||||
|
}
|
||||||
|
|
||||||
|
*i = append(*i, numbers...)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBindUnmarshalParamExtras(t *testing.T) {
|
||||||
|
// this test documents how bind handles `BindUnmarshaler` interface:
|
||||||
|
// NOTE: BindUnmarshaler chooses first input value to be bound.
|
||||||
|
|
||||||
|
t.Run("nok, unmarshalling fails", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?t=xxxx", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V unixTimestamp `query:"t"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.EqualError(t, err, "code=400, message='xxxx' is not an integer, internal='xxxx' is not an integer")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is struct", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?t=1710095540&t=1710095541", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V unixTimestamp `query:"t"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
expect := unixTimestamp{
|
||||||
|
Time: time.Unix(1710095540, 0),
|
||||||
|
}
|
||||||
|
assert.Equal(t, expect, result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is an alias to slice and is nil, append only values from first", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1,2,3&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V IntArrayA `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, IntArrayA([]int{1, 2, 3}), result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is an alias to slice and is nil, single input", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1,2", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V IntArrayA `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, IntArrayA([]int{1, 2}), result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is pointer an alias to slice and is nil", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V *IntArrayA `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
var expected = IntArrayA([]int{1})
|
||||||
|
assert.Equal(t, &expected, result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is pointer an alias to slice and is NOT nil", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V *IntArrayA `query:"a"`
|
||||||
|
}{}
|
||||||
|
result.V = new(IntArrayA) // NOT nil
|
||||||
|
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
var expected = IntArrayA([]int{1})
|
||||||
|
assert.Equal(t, &expected, result.V)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type unixTimestampLast struct {
|
||||||
|
Time time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
// this is silly example for `bindMultipleUnmarshaler` for type that uses last input value for unmarshalling
|
||||||
|
func (t *unixTimestampLast) UnmarshalParams(params []string) error {
|
||||||
|
lastInput := params[len(params)-1]
|
||||||
|
n, err := strconv.ParseInt(lastInput, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("'%s' is not an integer", lastInput)
|
||||||
|
}
|
||||||
|
*t = unixTimestampLast{Time: time.Unix(n, 0)}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
type IntArrayB []int
|
||||||
|
|
||||||
|
func (i *IntArrayB) UnmarshalParams(params []string) error {
|
||||||
|
var numbers = make([]int, 0, len(params))
|
||||||
|
|
||||||
|
for _, param := range params {
|
||||||
|
var values = strings.Split(param, ",")
|
||||||
|
for _, v := range values {
|
||||||
|
n, err := strconv.ParseInt(v, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("'%s' is not an integer", v)
|
||||||
|
}
|
||||||
|
numbers = append(numbers, int(n))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*i = append(*i, numbers...)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBindUnmarshalParams(t *testing.T) {
|
||||||
|
// this test documents how bind handles `bindMultipleUnmarshaler` interface:
|
||||||
|
|
||||||
|
t.Run("nok, unmarshalling fails", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?t=xxxx", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V unixTimestampLast `query:"t"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.EqualError(t, err, "code=400, message='xxxx' is not an integer, internal='xxxx' is not an integer")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is struct", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?t=1710095540&t=1710095541", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V unixTimestampLast `query:"t"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
expect := unixTimestampLast{
|
||||||
|
Time: time.Unix(1710095541, 0),
|
||||||
|
}
|
||||||
|
assert.Equal(t, expect, result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is an alias to slice and is nil, append multiple inputs", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1,2,3&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V IntArrayB `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, IntArrayB([]int{1, 2, 3, 4, 5, 6}), result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is an alias to slice and is nil, single input", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1,2", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V IntArrayB `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, IntArrayB([]int{1, 2}), result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is pointer an alias to slice and is nil", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V *IntArrayB `query:"a"`
|
||||||
|
}{}
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
var expected = IntArrayB([]int{1, 4, 5, 6})
|
||||||
|
assert.Equal(t, &expected, result.V)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ok, target is pointer an alias to slice and is NOT nil", func(t *testing.T) {
|
||||||
|
e := New()
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/?a=1&a=4,5,6", nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
c := e.NewContext(req, rec)
|
||||||
|
result := struct {
|
||||||
|
V *IntArrayB `query:"a"`
|
||||||
|
}{}
|
||||||
|
result.V = new(IntArrayB) // NOT nil
|
||||||
|
|
||||||
|
err := c.Bind(&result)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
var expected = IntArrayB([]int{1, 4, 5, 6})
|
||||||
|
assert.Equal(t, &expected, result.V)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user