mirror of
https://github.com/kataras/iris.git
synced 2025-12-19 02:47:04 +00:00
implement the makeHandler and structure the high-level mvc's API
Former-commit-id: 412118eae436981711ef57821f2d85b77a5d1a12
This commit is contained in:
@@ -4,13 +4,21 @@ import (
|
||||
"fmt"
|
||||
"reflect"
|
||||
|
||||
"github.com/kataras/golog"
|
||||
"github.com/kataras/iris/context"
|
||||
"github.com/kataras/iris/mvc/activator/methodfunc"
|
||||
)
|
||||
|
||||
// checks if "handler" is context.Handler; func(context.Context).
|
||||
func isContextHandler(handler interface{}) bool {
|
||||
_, is := handler.(context.Handler)
|
||||
return is
|
||||
func isContextHandler(handler interface{}) (context.Handler, bool) {
|
||||
h, is := handler.(context.Handler)
|
||||
if !is {
|
||||
fh, is := handler.(func(context.Context))
|
||||
if is {
|
||||
return fh, is
|
||||
}
|
||||
}
|
||||
return h, is
|
||||
}
|
||||
|
||||
func validateHandler(handler interface{}) error {
|
||||
@@ -19,3 +27,58 @@ func validateHandler(handler interface{}) error {
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
var (
|
||||
contextTyp = reflect.TypeOf(context.NewContext(nil))
|
||||
emptyIn = []reflect.Value{}
|
||||
)
|
||||
|
||||
func makeHandler(handler interface{}, binders []*InputBinder) context.Handler {
|
||||
if err := validateHandler(handler); err != nil {
|
||||
golog.Errorf("mvc handler: %v", err)
|
||||
return nil
|
||||
}
|
||||
|
||||
if h, is := isContextHandler(handler); is {
|
||||
golog.Warnf("mvc handler: you could just use the low-level API to register a context handler instead")
|
||||
return h
|
||||
}
|
||||
|
||||
typ := indirectTyp(reflect.TypeOf(handler))
|
||||
n := typ.NumIn()
|
||||
typIn := make([]reflect.Type, n, n)
|
||||
for i := 0; i < n; i++ {
|
||||
typIn[i] = typ.In(i)
|
||||
}
|
||||
|
||||
m := getBindersForInput(binders, typIn...)
|
||||
if len(m) != n {
|
||||
golog.Errorf("mvc handler: input arguments length(%d) and valid binders length(%d) are not equal", n, len(m))
|
||||
return nil
|
||||
}
|
||||
|
||||
hasIn := len(m) > 0
|
||||
fn := reflect.ValueOf(handler)
|
||||
|
||||
return func(ctx context.Context) {
|
||||
if !hasIn {
|
||||
methodfunc.DispatchFuncResult(ctx, fn.Call(emptyIn))
|
||||
return
|
||||
}
|
||||
|
||||
// we could use other tricks for "in"
|
||||
// here but let's stick to that which is clearly
|
||||
// that it doesn't keep any previous state
|
||||
// and it allocates exactly what we need,
|
||||
// so we can set via index instead of append.
|
||||
// The other method we could use is to
|
||||
// declare the in on the build state (before the return)
|
||||
// and use in[0:0] with append later on.
|
||||
in := make([]reflect.Value, n, n)
|
||||
ctxValues := []reflect.Value{reflect.ValueOf(ctx)}
|
||||
for k, v := range m {
|
||||
in[k] = v.BindFunc(ctxValues)
|
||||
}
|
||||
methodfunc.DispatchFuncResult(ctx, fn.Call(in))
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user