2015-05-17 22:54:29 -07:00
|
|
|
package middleware
|
|
|
|
|
|
|
|
import (
|
2016-03-14 13:55:38 -07:00
|
|
|
"fmt"
|
2022-03-16 00:29:42 +01:00
|
|
|
"net/http"
|
2016-03-14 13:55:38 -07:00
|
|
|
"runtime"
|
2015-05-17 22:54:29 -07:00
|
|
|
|
2019-01-30 12:56:56 +02:00
|
|
|
"github.com/labstack/echo/v4"
|
2020-07-06 23:59:42 +09:00
|
|
|
"github.com/labstack/gommon/log"
|
2015-05-17 22:54:29 -07:00
|
|
|
)
|
|
|
|
|
2016-02-17 21:01:47 -08:00
|
|
|
type (
|
2022-01-24 13:23:41 +03:00
|
|
|
// LogErrorFunc defines a function for custom logging in the middleware.
|
|
|
|
LogErrorFunc func(c echo.Context, err error, stack []byte) error
|
|
|
|
|
2016-07-27 09:34:44 -07:00
|
|
|
// RecoverConfig defines the config for Recover middleware.
|
2016-03-14 13:55:38 -07:00
|
|
|
RecoverConfig struct {
|
2016-07-27 09:34:44 -07:00
|
|
|
// Skipper defines a function to skip middleware.
|
|
|
|
Skipper Skipper
|
|
|
|
|
2016-05-10 11:52:04 -07:00
|
|
|
// Size of the stack to be printed.
|
|
|
|
// Optional. Default value 4KB.
|
2017-12-28 11:24:34 -08:00
|
|
|
StackSize int `yaml:"stack_size"`
|
2016-03-19 15:47:20 -07:00
|
|
|
|
2016-04-05 18:57:57 -07:00
|
|
|
// DisableStackAll disables formatting stack traces of all other goroutines
|
|
|
|
// into buffer after the trace for the current goroutine.
|
2016-05-10 11:52:04 -07:00
|
|
|
// Optional. Default value false.
|
2017-12-28 11:24:34 -08:00
|
|
|
DisableStackAll bool `yaml:"disable_stack_all"`
|
2016-03-19 15:47:20 -07:00
|
|
|
|
2016-04-05 18:57:57 -07:00
|
|
|
// DisablePrintStack disables printing stack trace.
|
2016-05-10 11:52:04 -07:00
|
|
|
// Optional. Default value as false.
|
2017-12-28 11:24:34 -08:00
|
|
|
DisablePrintStack bool `yaml:"disable_print_stack"`
|
2020-07-06 23:59:42 +09:00
|
|
|
|
|
|
|
// LogLevel is log level to printing stack trace.
|
|
|
|
// Optional. Default value 0 (Print).
|
|
|
|
LogLevel log.Lvl
|
2022-01-24 13:23:41 +03:00
|
|
|
|
|
|
|
// LogErrorFunc defines a function for custom logging in the middleware.
|
|
|
|
// If it's set you don't need to provide LogLevel for config.
|
|
|
|
LogErrorFunc LogErrorFunc
|
2016-02-17 21:01:47 -08:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2016-03-14 13:55:38 -07:00
|
|
|
var (
|
2016-07-27 09:34:44 -07:00
|
|
|
// DefaultRecoverConfig is the default Recover middleware config.
|
2016-03-14 13:55:38 -07:00
|
|
|
DefaultRecoverConfig = RecoverConfig{
|
2017-01-28 11:43:56 -08:00
|
|
|
Skipper: DefaultSkipper,
|
2016-04-05 18:57:57 -07:00
|
|
|
StackSize: 4 << 10, // 4 KB
|
|
|
|
DisableStackAll: false,
|
|
|
|
DisablePrintStack: false,
|
2020-07-06 23:59:42 +09:00
|
|
|
LogLevel: 0,
|
2022-01-24 13:23:41 +03:00
|
|
|
LogErrorFunc: nil,
|
2016-03-14 13:55:38 -07:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2016-03-19 15:47:20 -07:00
|
|
|
// Recover returns a middleware which recovers from panics anywhere in the chain
|
|
|
|
// and handles the control to the centralized HTTPErrorHandler.
|
2016-03-14 13:55:38 -07:00
|
|
|
func Recover() echo.MiddlewareFunc {
|
2016-04-07 21:20:50 -07:00
|
|
|
return RecoverWithConfig(DefaultRecoverConfig)
|
2016-03-14 13:55:38 -07:00
|
|
|
}
|
|
|
|
|
2016-08-31 20:10:14 -07:00
|
|
|
// RecoverWithConfig returns a Recover middleware with config.
|
2016-05-12 17:45:00 -07:00
|
|
|
// See: `Recover()`.
|
2016-04-07 21:20:50 -07:00
|
|
|
func RecoverWithConfig(config RecoverConfig) echo.MiddlewareFunc {
|
2016-03-31 16:30:19 -07:00
|
|
|
// Defaults
|
2016-07-27 09:34:44 -07:00
|
|
|
if config.Skipper == nil {
|
|
|
|
config.Skipper = DefaultRecoverConfig.Skipper
|
|
|
|
}
|
2016-03-31 16:30:19 -07:00
|
|
|
if config.StackSize == 0 {
|
|
|
|
config.StackSize = DefaultRecoverConfig.StackSize
|
|
|
|
}
|
|
|
|
|
2016-04-02 14:19:39 -07:00
|
|
|
return func(next echo.HandlerFunc) echo.HandlerFunc {
|
2017-09-21 12:07:04 -07:00
|
|
|
return func(c echo.Context) error {
|
2016-07-27 09:34:44 -07:00
|
|
|
if config.Skipper(c) {
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
|
2016-02-17 21:49:31 -08:00
|
|
|
defer func() {
|
2016-03-10 16:35:20 -08:00
|
|
|
if r := recover(); r != nil {
|
2022-03-16 00:29:42 +01:00
|
|
|
if r == http.ErrAbortHandler {
|
|
|
|
panic(r)
|
|
|
|
}
|
2017-09-21 12:07:04 -07:00
|
|
|
err, ok := r.(error)
|
|
|
|
if !ok {
|
2016-03-14 13:55:38 -07:00
|
|
|
err = fmt.Errorf("%v", r)
|
|
|
|
}
|
2022-01-24 13:23:41 +03:00
|
|
|
var stack []byte
|
|
|
|
var length int
|
|
|
|
|
2016-04-05 18:57:57 -07:00
|
|
|
if !config.DisablePrintStack {
|
2022-01-24 13:23:41 +03:00
|
|
|
stack = make([]byte, config.StackSize)
|
|
|
|
length = runtime.Stack(stack, !config.DisableStackAll)
|
|
|
|
stack = stack[:length]
|
|
|
|
}
|
|
|
|
|
|
|
|
if config.LogErrorFunc != nil {
|
|
|
|
err = config.LogErrorFunc(c, err, stack)
|
|
|
|
} else if !config.DisablePrintStack {
|
2020-07-06 23:59:42 +09:00
|
|
|
msg := fmt.Sprintf("[PANIC RECOVER] %v %s\n", err, stack[:length])
|
|
|
|
switch config.LogLevel {
|
|
|
|
case log.DEBUG:
|
|
|
|
c.Logger().Debug(msg)
|
|
|
|
case log.INFO:
|
|
|
|
c.Logger().Info(msg)
|
|
|
|
case log.WARN:
|
|
|
|
c.Logger().Warn(msg)
|
|
|
|
case log.ERROR:
|
|
|
|
c.Logger().Error(msg)
|
|
|
|
case log.OFF:
|
|
|
|
// None.
|
|
|
|
default:
|
|
|
|
c.Logger().Print(msg)
|
|
|
|
}
|
2016-03-10 16:35:20 -08:00
|
|
|
}
|
2016-03-14 13:55:38 -07:00
|
|
|
c.Error(err)
|
2016-02-17 21:49:31 -08:00
|
|
|
}
|
|
|
|
}()
|
2016-04-02 14:19:39 -07:00
|
|
|
return next(c)
|
|
|
|
}
|
2016-02-17 21:49:31 -08:00
|
|
|
}
|
2015-05-17 22:54:29 -07:00
|
|
|
}
|