2017-12-25 19:05:32 +01:00
|
|
|
package hero
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12/context"
|
2020-02-29 13:18:15 +01:00
|
|
|
)
|
2017-12-25 19:05:32 +01:00
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
type (
|
2020-03-01 07:34:53 +01:00
|
|
|
// ErrorHandler describes an interface to handle errors per hero handler and its dependencies.
|
|
|
|
//
|
|
|
|
// Handles non-nil errors return from a hero handler or a controller's method (see `getBindingsFor` and `Handler`)
|
|
|
|
// the error may return from a request-scoped dependency too (see `Handler`).
|
2020-02-29 13:18:15 +01:00
|
|
|
ErrorHandler interface {
|
|
|
|
HandleError(context.Context, error)
|
|
|
|
}
|
2020-03-01 07:34:53 +01:00
|
|
|
// ErrorHandlerFunc implements the `ErrorHandler`.
|
|
|
|
// It describes the type defnition for an error function handler.
|
2020-02-29 13:18:15 +01:00
|
|
|
ErrorHandlerFunc func(context.Context, error)
|
2017-12-25 19:05:32 +01:00
|
|
|
)
|
|
|
|
|
2020-03-01 07:34:53 +01:00
|
|
|
// HandleError fires when a non-nil error returns from a request-scoped dependency at serve-time or the handler itself.
|
2020-02-29 13:18:15 +01:00
|
|
|
func (fn ErrorHandlerFunc) HandleError(ctx context.Context, err error) {
|
|
|
|
fn(ctx, err)
|
|
|
|
}
|
2017-12-25 19:05:32 +01:00
|
|
|
|
2020-03-01 01:17:19 +01:00
|
|
|
var (
|
|
|
|
// ErrSeeOther may be returned from a dependency handler to skip a specific dependency
|
|
|
|
// based on custom logic.
|
|
|
|
ErrSeeOther = fmt.Errorf("see other")
|
|
|
|
// ErrStopExecution may be returned from a dependency handler to stop
|
|
|
|
// and return the execution of the function without error (it calls ctx.StopExecution() too).
|
|
|
|
// It may be occurred from request-scoped dependencies as well.
|
|
|
|
ErrStopExecution = fmt.Errorf("stop execution")
|
|
|
|
)
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
var (
|
|
|
|
// DefaultErrStatusCode is the default error status code (400)
|
|
|
|
// when the response contains a non-nil error or a request-scoped binding error occur.
|
|
|
|
DefaultErrStatusCode = 400
|
2020-02-14 22:34:56 +01:00
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
// DefaultErrorHandler is the default error handler which is fired
|
|
|
|
// when a function returns a non-nil error or a request-scoped dependency failed to binded.
|
|
|
|
DefaultErrorHandler = ErrorHandlerFunc(func(ctx context.Context, err error) {
|
2020-03-01 01:17:19 +01:00
|
|
|
if err != ErrStopExecution {
|
|
|
|
if status := ctx.GetStatusCode(); status == 0 || !context.StatusCodeNotSuccessful(status) {
|
|
|
|
ctx.StatusCode(DefaultErrStatusCode)
|
|
|
|
}
|
|
|
|
|
2020-03-01 07:34:53 +01:00
|
|
|
_, _ = ctx.WriteString(err.Error())
|
2020-02-29 13:18:15 +01:00
|
|
|
}
|
2017-12-25 19:05:32 +01:00
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
ctx.StopExecution()
|
|
|
|
})
|
|
|
|
)
|
|
|
|
|
|
|
|
func makeHandler(fn interface{}, c *Container) context.Handler {
|
|
|
|
if fn == nil {
|
|
|
|
panic("makeHandler: function is nil")
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
// 0. A normal handler.
|
|
|
|
if handler, ok := isHandler(fn); ok {
|
|
|
|
return handler
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
// 1. A handler which returns just an error, handle it faster.
|
|
|
|
if handlerWithErr, ok := isHandlerWithError(fn); ok {
|
|
|
|
return func(ctx context.Context) {
|
|
|
|
if err := handlerWithErr(ctx); err != nil {
|
|
|
|
c.GetErrorHandler(ctx).HandleError(ctx, err)
|
|
|
|
}
|
|
|
|
}
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
v := valueOf(fn)
|
|
|
|
numIn := v.Type().NumIn()
|
2017-12-25 19:05:32 +01:00
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
bindings := getBindingsForFunc(v, c.Dependencies, c.ParamStartIndex)
|
|
|
|
|
|
|
|
return func(ctx context.Context) {
|
|
|
|
inputs := make([]reflect.Value, numIn)
|
|
|
|
|
|
|
|
for _, binding := range bindings {
|
|
|
|
input, err := binding.Dependency.Handle(ctx, binding.Input)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrSeeOther {
|
|
|
|
continue
|
|
|
|
}
|
2020-03-01 01:17:19 +01:00
|
|
|
// handled inside ErrorHandler.
|
|
|
|
// else if err == ErrStopExecution {
|
|
|
|
// ctx.StopExecution()
|
|
|
|
// return // return without error.
|
|
|
|
// }
|
2020-02-29 13:18:15 +01:00
|
|
|
|
|
|
|
c.GetErrorHandler(ctx).HandleError(ctx, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
inputs[binding.Input.Index] = input
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
outputs := v.Call(inputs)
|
|
|
|
if err := dispatchFuncResult(ctx, outputs); err != nil {
|
|
|
|
c.GetErrorHandler(ctx).HandleError(ctx, err)
|
|
|
|
}
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
2020-02-29 13:18:15 +01:00
|
|
|
}
|
2017-12-25 19:05:32 +01:00
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
func isHandler(fn interface{}) (context.Handler, bool) {
|
|
|
|
if handler, ok := fn.(context.Handler); ok {
|
|
|
|
return handler, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
if handler, ok := fn.(func(context.Context)); ok {
|
|
|
|
return handler, ok
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func isHandlerWithError(fn interface{}) (func(context.Context) error, bool) {
|
|
|
|
if handlerWithErr, ok := fn.(func(context.Context) error); ok {
|
|
|
|
return handlerWithErr, true
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|
|
|
|
|
2020-02-29 13:18:15 +01:00
|
|
|
return nil, false
|
2017-12-25 19:05:32 +01:00
|
|
|
}
|