2017-11-23 21:36:47 +01:00
|
|
|
package mvc2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
2017-12-10 06:00:51 +01:00
|
|
|
"runtime"
|
2017-11-23 21:36:47 +01:00
|
|
|
|
2017-11-24 11:32:35 +01:00
|
|
|
"github.com/kataras/golog"
|
2017-11-23 21:36:47 +01:00
|
|
|
"github.com/kataras/iris/context"
|
|
|
|
)
|
|
|
|
|
|
|
|
// checks if "handler" is context.Handler; func(context.Context).
|
2017-11-24 11:32:35 +01:00
|
|
|
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
|
2017-11-23 21:36:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func validateHandler(handler interface{}) error {
|
|
|
|
if typ := reflect.TypeOf(handler); !isFunc(typ) {
|
|
|
|
return fmt.Errorf("handler expected to be a kind of func but got typeof(%s)", typ.String())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2017-11-24 11:32:35 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
// MustMakeHandler calls the `MakeHandler` and panics on any error.
|
|
|
|
func MustMakeHandler(handler interface{}, bindValues ...reflect.Value) context.Handler {
|
|
|
|
h, err := MakeHandler(handler, bindValues...)
|
2017-11-24 14:10:30 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2017-12-10 06:00:51 +01:00
|
|
|
|
2017-11-24 14:10:30 +01:00
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
// MakeHandler accepts a "handler" function which can accept any input that matches
|
|
|
|
// with the "binders" and any output, that matches the mvc types, like string, int (string,int),
|
|
|
|
// custom structs, Result(View | Response) and anything that you already know that mvc implementation supports,
|
|
|
|
// and returns a low-level `context/iris.Handler` which can be used anywhere in the Iris Application,
|
|
|
|
// as middleware or as simple route handler or party handler or subdomain handler-router.
|
2017-12-10 06:00:51 +01:00
|
|
|
func MakeHandler(handler interface{}, bindValues ...reflect.Value) (context.Handler, error) {
|
2017-11-24 11:32:35 +01:00
|
|
|
if err := validateHandler(handler); err != nil {
|
2017-11-24 14:10:30 +01:00
|
|
|
return nil, err
|
2017-11-24 11:32:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if h, is := isContextHandler(handler); is {
|
|
|
|
golog.Warnf("mvc handler: you could just use the low-level API to register a context handler instead")
|
2017-11-24 14:10:30 +01:00
|
|
|
return h, nil
|
2017-11-24 11:32:35 +01:00
|
|
|
}
|
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
fn := reflect.ValueOf(handler)
|
|
|
|
n := fn.Type().NumIn()
|
2017-11-24 11:32:35 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
if n == 0 {
|
|
|
|
h := func(ctx context.Context) {
|
|
|
|
DispatchFuncResult(ctx, fn.Call(emptyIn))
|
|
|
|
}
|
2017-11-25 13:04:35 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
return h, nil
|
|
|
|
}
|
2017-12-04 04:06:03 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
s := newTargetFunc(fn, bindValues...)
|
|
|
|
if !s.Valid {
|
|
|
|
pc := fn.Pointer()
|
|
|
|
fpc := runtime.FuncForPC(pc)
|
|
|
|
callerFileName, callerLineNumber := fpc.FileLine(pc)
|
|
|
|
callerName := fpc.Name()
|
2017-12-04 04:06:03 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
err := fmt.Errorf("input arguments length(%d) and valid binders length(%d) are not equal for typeof '%s' which is defined at %s:%d by %s",
|
|
|
|
n, len(s.Inputs), fn.Type().String(), callerFileName, callerLineNumber, callerName)
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-11-24 11:32:35 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
h := func(ctx context.Context) {
|
|
|
|
in := make([]reflect.Value, n, n)
|
2017-11-24 11:32:35 +01:00
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
s.Fill(&in, reflect.ValueOf(ctx))
|
|
|
|
if ctx.IsStopped() {
|
|
|
|
return
|
2017-12-04 04:06:03 +01:00
|
|
|
}
|
2017-12-10 06:00:51 +01:00
|
|
|
DispatchFuncResult(ctx, fn.Call(in))
|
2017-12-04 04:06:03 +01:00
|
|
|
}
|
|
|
|
|
2017-12-10 06:00:51 +01:00
|
|
|
return h, nil
|
2017-11-24 11:32:35 +01:00
|
|
|
|
|
|
|
}
|