mirror of
https://github.com/kataras/iris.git
synced 2025-01-24 19:21:03 +01:00
5a3be2ab58
Former-commit-id: 81ae99390c683a61e1b0bac58725a04b9a3eebbb
154 lines
4.0 KiB
Go
154 lines
4.0 KiB
Go
package mvc2
|
|
|
|
import (
|
|
"reflect"
|
|
)
|
|
|
|
// InputBinder is the result of `MakeBinder`.
|
|
// It contains the binder wrapped information, like the
|
|
// type that is responsible to bind
|
|
// and a function which will accept a context and returns a value of something.
|
|
type InputBinder struct {
|
|
BindType reflect.Type
|
|
// ctx is slice because all binder functions called by
|
|
// their `.Call` method which accepts a slice of reflect.Value,
|
|
// so on the handler maker we will allocate a slice of a single ctx once
|
|
// and used to all binders.
|
|
BindFunc func(ctx []reflect.Value) reflect.Value
|
|
}
|
|
|
|
// getBindersForInput returns a map of the responsible binders for the "expected" types,
|
|
// which are the expected input parameters' types,
|
|
// based on the available "binders" collection.
|
|
//
|
|
// It returns a map which its key is the index of the "expected" which
|
|
// a valid binder for that in's type found,
|
|
// the value is the pointer of the responsible `InputBinder`.
|
|
//
|
|
// Check of "a nothing responsible for those expected types"
|
|
// should be done using the `len(m) == 0`.
|
|
func getBindersForInput(binders []*InputBinder, expected ...reflect.Type) map[int]*InputBinder {
|
|
var m map[int]*InputBinder
|
|
|
|
for idx, in := range expected {
|
|
if idx == 0 && isContext(in) {
|
|
// if the first is context then set it directly here.
|
|
m = make(map[int]*InputBinder)
|
|
m[0] = &InputBinder{
|
|
BindType: contextTyp,
|
|
BindFunc: func(ctxValues []reflect.Value) reflect.Value {
|
|
return ctxValues[0]
|
|
},
|
|
}
|
|
continue
|
|
}
|
|
for _, b := range binders {
|
|
// if same type or the result of binder implements the expected in's type.
|
|
if equalTypes(b.BindType, in) {
|
|
if m == nil {
|
|
m = make(map[int]*InputBinder)
|
|
}
|
|
// fmt.Printf("set index: %d to type: %s where input type is: %s\n", idx, b.BindType.String(), in.String())
|
|
m[idx] = b
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
return m
|
|
}
|
|
|
|
// MustMakeFuncInputBinder calls the `MakeFuncInputBinder` and returns its first result, see its docs.
|
|
// It panics on error.
|
|
func MustMakeFuncInputBinder(binder interface{}) *InputBinder {
|
|
b, err := MakeFuncInputBinder(binder)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return b
|
|
}
|
|
|
|
type binderType uint32
|
|
|
|
const (
|
|
functionType binderType = iota
|
|
serviceType
|
|
invalidType
|
|
)
|
|
|
|
func resolveBinderType(binder interface{}) binderType {
|
|
if binder == nil {
|
|
return invalidType
|
|
}
|
|
|
|
switch indirectTyp(reflect.TypeOf(binder)).Kind() {
|
|
case reflect.Func:
|
|
return functionType
|
|
case reflect.Struct:
|
|
return serviceType
|
|
}
|
|
|
|
return invalidType
|
|
}
|
|
|
|
// MakeFuncInputBinder takes a binder function or a struct which contains a "Bind"
|
|
// function and returns an `InputBinder`, which Iris uses to
|
|
// resolve and set the input parameters when a handler is executed.
|
|
//
|
|
// The "binder" can have the following form:
|
|
// `func(iris.Context) UserViewModel`.
|
|
//
|
|
// The return type of the "binder" should be a value instance, not a pointer, for your own protection.
|
|
// The binder function should return only one value and
|
|
// it can accept only one input argument, the Iris' Context (`context.Context` or `iris.Context`).
|
|
func MakeFuncInputBinder(binder interface{}) (*InputBinder, error) {
|
|
v := reflect.ValueOf(binder)
|
|
return makeFuncInputBinder(v)
|
|
}
|
|
|
|
func makeFuncInputBinder(fn reflect.Value) (*InputBinder, error) {
|
|
typ := indirectTyp(fn.Type())
|
|
|
|
// invalid if not a func.
|
|
if typ.Kind() != reflect.Func {
|
|
return nil, errBad
|
|
}
|
|
|
|
// invalid if not returns one single value.
|
|
if typ.NumOut() != 1 {
|
|
return nil, errBad
|
|
}
|
|
|
|
// invalid if input args length is not one.
|
|
if typ.NumIn() != 1 {
|
|
return nil, errBad
|
|
}
|
|
|
|
// invalid if that single input arg is not a typeof context.Context.
|
|
if !isContext(typ.In(0)) {
|
|
return nil, errBad
|
|
}
|
|
|
|
outTyp := typ.Out(0)
|
|
zeroOutVal := reflect.New(outTyp).Elem()
|
|
|
|
bf := func(ctxValue []reflect.Value) reflect.Value {
|
|
// []reflect.Value{reflect.ValueOf(ctx)}
|
|
results := fn.Call(ctxValue)
|
|
if len(results) == 0 {
|
|
return zeroOutVal
|
|
}
|
|
|
|
v := results[0]
|
|
if !v.IsValid() {
|
|
return zeroOutVal
|
|
}
|
|
return v
|
|
}
|
|
|
|
return &InputBinder{
|
|
BindType: outTyp,
|
|
BindFunc: bf,
|
|
}, nil
|
|
}
|