black-box the MakeHandler, works perfectly.

Former-commit-id: d325be0e953efc2f841c69f62233b34d4a58ab62
This commit is contained in:
kataras 2017-11-24 15:10:30 +02:00
parent bfec1d174f
commit 29835d9a8e
5 changed files with 126 additions and 51 deletions

View File

@ -33,15 +33,30 @@ var (
emptyIn = []reflect.Value{} emptyIn = []reflect.Value{}
) )
func makeHandler(handler interface{}, binders []*InputBinder) context.Handler { // MustMakeHandler calls the `MakeHandler` and returns its first resultthe low-level handler), see its docs.
// It panics on error.
func MustMakeHandler(handler interface{}, binders []*InputBinder) context.Handler {
h, err := MakeHandler(handler, binders)
if err != nil {
panic(err)
}
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.
func MakeHandler(handler interface{}, binders []*InputBinder) (context.Handler, error) {
if err := validateHandler(handler); err != nil { if err := validateHandler(handler); err != nil {
golog.Errorf("mvc handler: %v", err) golog.Errorf("mvc handler: %v", err)
return nil return nil, err
} }
if h, is := isContextHandler(handler); is { if h, is := isContextHandler(handler); is {
golog.Warnf("mvc handler: you could just use the low-level API to register a context handler instead") golog.Warnf("mvc handler: you could just use the low-level API to register a context handler instead")
return h return h, nil
} }
typ := indirectTyp(reflect.TypeOf(handler)) typ := indirectTyp(reflect.TypeOf(handler))
@ -53,14 +68,15 @@ func makeHandler(handler interface{}, binders []*InputBinder) context.Handler {
m := getBindersForInput(binders, typIn...) m := getBindersForInput(binders, typIn...)
if len(m) != n { if len(m) != n {
golog.Errorf("mvc handler: input arguments length(%d) and valid binders length(%d) are not equal", n, len(m)) err := fmt.Errorf("input arguments length(%d) of types(%s) and valid binders length(%d) are not equal", n, typIn, len(m))
return nil golog.Errorf("mvc handler: %v", err)
return nil, err
} }
hasIn := len(m) > 0 hasIn := len(m) > 0
fn := reflect.ValueOf(handler) fn := reflect.ValueOf(handler)
return func(ctx context.Context) { resultHandler := func(ctx context.Context) {
if !hasIn { if !hasIn {
methodfunc.DispatchFuncResult(ctx, fn.Call(emptyIn)) methodfunc.DispatchFuncResult(ctx, fn.Call(emptyIn))
return return
@ -81,4 +97,6 @@ func makeHandler(handler interface{}, binders []*InputBinder) context.Handler {
} }
methodfunc.DispatchFuncResult(ctx, fn.Call(in)) methodfunc.DispatchFuncResult(ctx, fn.Call(in))
} }
return resultHandler, nil
} }

View File

@ -1,11 +1,96 @@
package mvc2 package mvc2_test
/* // black-box
TODO:
Test that as we test the rest, with import (
a fake context, and after move again to "fmt"
the mvc_test.go which will contain "testing"
the overall high-level (black-box) tests.
*/ "github.com/kataras/iris"
"github.com/kataras/iris/httptest"
. "github.com/kataras/iris/mvc2"
)
// dynamic func
type testUserStruct struct {
ID int64
Username string
}
func testBinderFunc(ctx iris.Context) testUserStruct {
id, _ := ctx.Params().GetInt64("id")
username := ctx.Params().Get("username")
return testUserStruct{
ID: id,
Username: username,
}
}
// service
type (
testService interface {
Say(string) string
}
testServiceImpl struct {
prefix string
}
)
func (s *testServiceImpl) Say(message string) string {
return s.prefix + " " + message
}
func TestMakeHandler(t *testing.T) {
binders := []*InputBinder{
// #1
MustMakeFuncInputBinder(testBinderFunc),
// #2
MustMakeServiceInputBinder(&testServiceImpl{prefix: "say"}),
// #3
MustMakeFuncInputBinder(func(ctx iris.Context) string {
return ctx.Params().Get("param")
}),
}
var (
// a context as first input arg, which is not needed to be binded manually,
// and a user struct which is binded to the input arg by the #1 func(ctx) any binder.
consumeUserHandler = func(ctx iris.Context, user testUserStruct) {
ctx.JSON(user)
}
h1 = MustMakeHandler(consumeUserHandler, binders)
// just one input arg, the service which is binded by the #2 service binder.
consumeServiceHandler = func(service testService) string {
return service.Say("something")
}
h2 = MustMakeHandler(consumeServiceHandler, binders)
// just one input arg, a standar string which is binded by the #3 func(ctx) any binder.
consumeParamHandler = func(myParam string) string {
return "param is: " + myParam
}
h3 = MustMakeHandler(consumeParamHandler, binders)
)
app := iris.New()
app.Get("/{id:long}/{username:string}", h1)
app.Get("/service", h2)
app.Get("/param/{param:string}", h3)
expectedUser := testUserStruct{
ID: 42,
Username: "kataras",
}
e := httptest.New(t, app)
// 1
e.GET(fmt.Sprintf("/%d/%s", expectedUser.ID, expectedUser.Username)).Expect().Status(httptest.StatusOK).
JSON().Equal(expectedUser)
// 2
e.GET("/service").Expect().Status(httptest.StatusOK).
Body().Equal("say something")
// 3
e.GET("/param/the_param_value").Expect().Status(httptest.StatusOK).
Body().Equal("param is: the_param_value")
}

View File

@ -45,5 +45,6 @@ func (m *Mvc) RegisterService(services ...interface{}) error {
} }
func (m *Mvc) Handler(handler interface{}) context.Handler { func (m *Mvc) Handler(handler interface{}) context.Handler {
return makeHandler(handler, m.binders) h, _ := MakeHandler(handler, m.binders) // it logs errors already, so on any error the "h" will be nil.
return h
} }

View File

@ -1,30 +0,0 @@
package mvc2_test
import (
"testing"
"github.com/kataras/iris"
"github.com/kataras/iris/httptest"
"github.com/kataras/iris/mvc2"
)
var (
lowLevelHandler = func(ctx iris.Context) {
ctx.Writef("low-level handler")
}
)
func TestHandler(t *testing.T) {
app := iris.New()
m := mvc2.New()
// should just return a context.Handler
// without performance penalties.
app.Get("/", m.Handler(lowLevelHandler))
e := httptest.New(t, app, httptest.LogLevel("debug"))
// 1
e.GET("/").Expect().Status(httptest.StatusOK).
Body().Equal("low-level handler")
}

View File

@ -33,14 +33,15 @@ func isFunc(typ reflect.Type) bool {
return typ.Kind() == reflect.Func return typ.Kind() == reflect.Func
} }
func equalTypes(in reflect.Type, v reflect.Type) bool { func equalTypes(got reflect.Type, expected reflect.Type) bool {
if in == v { if got == expected {
return true return true
} }
// if accepts an interface, check if the given "v" type does // if accepts an interface, check if the given "got" type does
// implement this. // implement this "expected" user handler's input argument.
if in.Kind() == reflect.Interface { if expected.Kind() == reflect.Interface {
return v.Implements(in) // fmt.Printf("expected interface = %s and got to set on the arg is: %s\n", expected.String(), got.String())
return got.Implements(expected)
} }
return false return false
} }