mirror of
https://github.com/kataras/iris.git
synced 2025-03-14 08:16:28 +01:00
black-box the MakeHandler, works perfectly.
Former-commit-id: d325be0e953efc2f841c69f62233b34d4a58ab62
This commit is contained in:
parent
bfec1d174f
commit
29835d9a8e
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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")
|
||||||
|
}
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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")
|
|
||||||
|
|
||||||
}
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user