Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: [#521] User can custom recover when a request panic #103

Merged
merged 87 commits into from
Dec 27, 2024
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
87 commits
Select commit Hold shift + click to select a range
e06c940
[Feature] User can custom recover when a request panic
KlassnayaAfrodita Nov 15, 2024
09245fe
Update middleware_timeout.go
KlassnayaAfrodita Nov 15, 2024
5168d1e
Update route.go
KlassnayaAfrodita Nov 15, 2024
32ee690
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 15, 2024
29d36ee
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 15, 2024
88daede
Update context_request.go
KlassnayaAfrodita Nov 15, 2024
e82b340
Update view.go
KlassnayaAfrodita Nov 15, 2024
f3efb50
Update gin.go
KlassnayaAfrodita Nov 15, 2024
61b20aa
Update context_request_test.go
KlassnayaAfrodita Nov 15, 2024
a771bce
Update service_provider.go
KlassnayaAfrodita Nov 15, 2024
3b77bd0
Update view.go
KlassnayaAfrodita Nov 15, 2024
cc7a261
Update gin.go
KlassnayaAfrodita Nov 15, 2024
21125ef
Update context_request_test.go
KlassnayaAfrodita Nov 16, 2024
f3c666a
Update context_request.go
KlassnayaAfrodita Nov 18, 2024
adf6a8c
Update context_request_test.go
KlassnayaAfrodita Nov 18, 2024
0a3998c
Update gin.go
KlassnayaAfrodita Nov 18, 2024
d001e4d
Update middleware_timeout.go
KlassnayaAfrodita Nov 18, 2024
28af043
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 18, 2024
b0a62cd
Update route.go
KlassnayaAfrodita Nov 18, 2024
59ae234
Update service_provider.go
KlassnayaAfrodita Nov 18, 2024
5745c8f
Update view.go
KlassnayaAfrodita Nov 18, 2024
fae6fbe
Update view.go
KlassnayaAfrodita Nov 18, 2024
98f64a5
Update route.go
KlassnayaAfrodita Nov 18, 2024
076581a
Update route.go
KlassnayaAfrodita Nov 21, 2024
d5d8a93
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
998e3c4
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
9a3074c
Update route.go
KlassnayaAfrodita Nov 21, 2024
a019972
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
3f2edef
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
f7a0ac6
Update context_request.go
KlassnayaAfrodita Nov 21, 2024
36978b1
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
92c38d4
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 21, 2024
f66dadd
Update context_request_test.go
KlassnayaAfrodita Nov 21, 2024
3915909
Update middleware_timeout.go
KlassnayaAfrodita Nov 21, 2024
faf902e
Update gin.go
KlassnayaAfrodita Nov 22, 2024
cf0b756
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 22, 2024
0023e51
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 22, 2024
3c5f3bc
Update view.go
KlassnayaAfrodita Nov 26, 2024
fb1faf6
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 26, 2024
8f27f62
Update middleware_timeout_test.go
KlassnayaAfrodita Nov 26, 2024
49f6b21
Update route.go
KlassnayaAfrodita Dec 13, 2024
0e34013
Update middleware_timeout.go
KlassnayaAfrodita Dec 13, 2024
99e06c8
Merge branch 'master' into patch-3
KlassnayaAfrodita Dec 13, 2024
c6d79c8
Update middleware_timeout.go
KlassnayaAfrodita Dec 13, 2024
e0727a7
Update context_request.go
KlassnayaAfrodita Dec 13, 2024
8a0c44e
Update context_request.go
KlassnayaAfrodita Dec 17, 2024
173e471
Update middleware_timeout.go
KlassnayaAfrodita Dec 17, 2024
e4a1360
Update service_provider.go
KlassnayaAfrodita Dec 17, 2024
f30e2f4
Update middleware_timeout.go
KlassnayaAfrodita Dec 17, 2024
6217114
Update route.go
KlassnayaAfrodita Dec 17, 2024
35e0182
Update view.go
KlassnayaAfrodita Dec 17, 2024
ace488a
Update gin.go
KlassnayaAfrodita Dec 17, 2024
f7f209e
Update route.go
KlassnayaAfrodita Dec 17, 2024
440ba87
Update context_request.go
KlassnayaAfrodita Dec 17, 2024
ca0c57f
Update service_provider.go
KlassnayaAfrodita Dec 17, 2024
4e8ddfb
Update route.go
KlassnayaAfrodita Dec 17, 2024
dfa4bd0
Update route.go
KlassnayaAfrodita Dec 17, 2024
44b41f6
Update route.go
KlassnayaAfrodita Dec 17, 2024
4346508
Update route.go
KlassnayaAfrodita Dec 17, 2024
1682fe7
Update middleware_timeout.go
KlassnayaAfrodita Dec 17, 2024
60320c1
Merge branch 'master' into patch-3
KlassnayaAfrodita Dec 17, 2024
28dcae8
Update view.go
KlassnayaAfrodita Dec 18, 2024
4aceba8
Update middleware_timeout.go
KlassnayaAfrodita Dec 19, 2024
601dbd0
Update middleware_timeout.go
KlassnayaAfrodita Dec 19, 2024
a79cae7
Update middleware_timeout.go
KlassnayaAfrodita Dec 19, 2024
a581045
Update middleware_timeout.go
KlassnayaAfrodita Dec 19, 2024
5a0101e
Update middleware_timeout.go
KlassnayaAfrodita Dec 19, 2024
1c07943
Update middleware_timeout_test.go
KlassnayaAfrodita Dec 19, 2024
f020930
Update middleware_timeout_test.go
KlassnayaAfrodita Dec 19, 2024
a0b8834
Update view.go
KlassnayaAfrodita Dec 19, 2024
47c53e7
Update view.go
KlassnayaAfrodita Dec 19, 2024
343d493
Update gin.go
KlassnayaAfrodita Dec 23, 2024
41d57a1
Update route.go
KlassnayaAfrodita Dec 23, 2024
057f865
Update service_provider.go
KlassnayaAfrodita Dec 23, 2024
4d82e79
Update view.go
KlassnayaAfrodita Dec 23, 2024
e6fb55f
Update route.go
KlassnayaAfrodita Dec 23, 2024
5df7701
Update route_test.go
KlassnayaAfrodita Dec 24, 2024
aa56ca1
Update route_test.go
KlassnayaAfrodita Dec 24, 2024
66e139e
Update route.go
KlassnayaAfrodita Dec 24, 2024
46fc772
Update route_test.go
KlassnayaAfrodita Dec 26, 2024
48144d5
Update route_test.go
KlassnayaAfrodita Dec 26, 2024
72a5064
Update route.go
KlassnayaAfrodita Dec 26, 2024
e7eef7b
Update route.go
KlassnayaAfrodita Dec 26, 2024
ee526b3
Update route_test.go
KlassnayaAfrodita Dec 26, 2024
4d7e660
Update route_test.go
KlassnayaAfrodita Dec 26, 2024
5371bd7
Update middleware_timeout_test.go
KlassnayaAfrodita Dec 26, 2024
cdf7d2f
Merge branch 'master' into patch-3
hwbrzzl Dec 26, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 5 additions & 2 deletions context_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,9 +34,12 @@ type ContextRequest struct {
func NewContextRequest(ctx *Context, log log.Log, validation contractsvalidate.Validation) contractshttp.ContextRequest {
httpBody, err := getHttpBody(ctx)
if err != nil {
LogFacade.Error(fmt.Sprintf("%+v", errors.Unwrap(err)))
if LogFacade != nil {
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
LogFacade.Error(fmt.Sprintf("%+v", errors.Unwrap(err)))
} else {
fmt.Println("LogFacade is nil, error:", fmt.Sprintf("%+v", errors.Unwrap(err)))
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
}
}

return &ContextRequest{ctx: ctx, instance: ctx.instance, httpBody: httpBody, log: log, validation: validation}
}

Expand Down
3 changes: 3 additions & 0 deletions context_request_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,9 @@

var err error
s.route, err = NewRoute(s.mockConfig, nil)
if s.route == nil {
fmt.Println("NewRoute returned nil")
}
s.Require().Nil(err)
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
}

Expand All @@ -53,7 +56,7 @@
}

func (s *ContextRequestSuite) TestAll() {
s.route.Get("/all", func(ctx contractshttp.Context) contractshttp.Response {

Check failure on line 59 in context_request_test.go

View workflow job for this annotation

GitHub Actions / lint / nilaway

error: Potential nil panic detected. Observed nil flow from source to dereference point:
return ctx.Response().Success().Json(contractshttp.Json{
"all": ctx.Request().All(),
})
Expand Down
22 changes: 14 additions & 8 deletions facades/gin.go
Original file line number Diff line number Diff line change
@@ -1,21 +1,27 @@
package facades

import (
"fmt"
"log"

"github.com/goravel/framework/contracts/route"

hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
"github.com/goravel/gin"
)

func Route(driver string) route.Route {
instance, err := gin.App.MakeWith(gin.RouteBinding, map[string]any{
"driver": driver,
})
if err != nil {
log.Fatalln(err)
if gin.App != nil {
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
instance, err := gin.App.MakeWith(gin.RouteBinding, map[string]any{
"driver": driver,
})

if err != nil {
log.Fatalln(err)
return nil
}

return instance.(*gin.Route)
} else {
fmt.Println("App is nil")
return nil
}

return instance.(*gin.Route)
}
15 changes: 9 additions & 6 deletions middleware_timeout.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import (
)

// Timeout creates middleware to set a timeout for a request
func Timeout(timeout time.Duration) contractshttp.Middleware {
func Timeout(timeout time.Duration, recoverFunc func(ctx contractshttp.Context, err interface{})) contractshttp.Middleware {
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
return func(ctx contractshttp.Context) {
timeoutCtx, cancel := context.WithTimeout(ctx.Context(), timeout)
defer cancel()
Expand All @@ -22,12 +22,15 @@ func Timeout(timeout time.Duration) contractshttp.Middleware {
go func() {
defer func() {
if r := recover(); r != nil {
if LogFacade != nil {
LogFacade.Request(ctx.Request()).Error(r)
// Use the custom recoverFunc if provided, otherwise fallback to default behavior
if recoverFunc != nil {
recoverFunc(ctx, r)
} else {
if LogFacade != nil {
LogFacade.Request(ctx.Request()).Error(r)
}
_ = ctx.Response().Status(http.StatusInternalServerError).String("Internal Server Error").Render()
}

// TODO can be customized in https://github.com/goravel/goravel/issues/521
_ = ctx.Response().Status(http.StatusInternalServerError).String("Internal Server Error").Render()
}

close(done)
Expand Down
8 changes: 4 additions & 4 deletions middleware_timeout_test.go
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add a test case to test the custom recover as well.

Original file line number Diff line number Diff line change
Expand Up @@ -14,23 +14,23 @@ import (
"github.com/stretchr/testify/require"
)

func TestTimeoutMiddleware(t *testing.T) {
func TestTmeoutMiddleware(t *testing.T) {
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
mockConfig := mocksconfig.NewConfig(t)
mockConfig.EXPECT().GetBool("app.debug").Return(true).Once()
mockConfig.EXPECT().GetInt("http.drivers.gin.body_limit", 4096).Return(4096).Once()

route, err := NewRoute(mockConfig, nil)
require.NoError(t, err)

route.Middleware(Timeout(1*time.Second)).Get("/timeout", func(ctx contractshttp.Context) contractshttp.Response {
route.Middleware(Timeout(1*time.Second, nil)).Get("/timeout", func(ctx contractshttp.Context) contractshttp.Response {
time.Sleep(2 * time.Second)

return ctx.Response().Success().String("timeout")
})
route.Middleware(Timeout(1*time.Second)).Get("/normal", func(ctx contractshttp.Context) contractshttp.Response {
route.Middleware(Timeout(1*time.Second, nil)).Get("/normal", func(ctx contractshttp.Context) contractshttp.Response {
return ctx.Response().Success().String("normal")
})
route.Middleware(Timeout(1*time.Second)).Get("/panic", func(ctx contractshttp.Context) contractshttp.Response {
route.Middleware(Timeout(1*time.Second, nil)).Get("/panic", func(ctx contractshttp.Context) contractshttp.Response {
panic(1)
})

Expand Down
17 changes: 16 additions & 1 deletion route.go
Original file line number Diff line number Diff line change
Expand Up @@ -82,12 +82,27 @@ func (r *Route) Fallback(handler httpcontract.HandlerFunc) {
func (r *Route) GlobalMiddleware(middlewares ...httpcontract.Middleware) {
timeout := time.Duration(r.config.GetInt("http.request_timeout", 3)) * time.Second
defaultMiddlewares := []httpcontract.Middleware{
Cors(), Tls(), Timeout(timeout),
Cors(), Tls(), Timeout(timeout, nil),
}
middlewares = append(defaultMiddlewares, middlewares...)
r.setMiddlewares(middlewares)
}

func (r *Route) Recover(recoverFunc func(ctx *gin.Context, err interface{})) {
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
r.instance.Use(func(ctx *gin.Context) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is a CustomRecovery method in Gin, we can use it directly.

image

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

will we end up using gin.CustomRecovery?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If it can be used directly, yes, we can.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is CustomRecovery suitable, please?

defer func() {
if err := recover(); err != nil {
if recoverFunc != nil {
recoverFunc(ctx, err)
} else {
ctx.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"error": "Internal Server Error"})
}
}
}()
ctx.Next()
})
}

func (r *Route) Run(host ...string) error {
if len(host) == 0 {
defaultHost := r.config.GetString("http.host")
Expand Down
11 changes: 11 additions & 0 deletions service_provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (
"github.com/goravel/framework/contracts/validation"
"github.com/goravel/framework/errors"
"github.com/goravel/framework/support/color"
"os"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sort it:

system packages

third packages

internal packages

)

const RouteBinding = "goravel.gin.route"
Expand Down Expand Up @@ -35,13 +36,18 @@ func (receiver *ServiceProvider) Boot(app foundation.Application) {

if ConfigFacade = app.MakeConfig(); ConfigFacade == nil {
color.Errorln(errors.ConfigFacadeNotSet.SetModule(module))
shutdownOnCriticalError("ConfigFacade is not set")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's unnecessary for now, we want to print a warning instead of interrupting.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to remove the function totally.

}

if LogFacade = app.MakeLog(); LogFacade == nil {
color.Errorln(errors.LogFacadeNotSet.SetModule(module))
shutdownOnCriticalError("LogFacade is not set")
}

if ValidationFacade = app.MakeValidation(); ValidationFacade == nil {
color.Errorln(errors.New("validation facade is not initialized").SetModule(module))
}

if ViewFacade = app.MakeView(); ViewFacade == nil {
color.Errorln(errors.New("view facade is not initialized").SetModule(module))
}
Expand All @@ -50,3 +56,8 @@ func (receiver *ServiceProvider) Boot(app foundation.Application) {
"config/cors.go": app.ConfigPath("cors.go"),
})
}

func shutdownOnCriticalError(message string) {
color.Errorln("Critical error:", message)
os.Exit(1) // Force exit to ensure the application does not run with missing dependencies
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove it, it's unnecessary.

17 changes: 13 additions & 4 deletions view.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,13 @@
}

func (receive *View) Make(view string, data ...any) contractshttp.Response {
shared := ViewFacade.GetShared()
var shared map[string]any
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
if ViewFacade != nil {
shared = ViewFacade.GetShared()
} else {
fmt.Println("ViewFacade is nil")
}

hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
if len(data) == 0 {
return &HtmlResponse{shared, receive.instance, view}
} else {
Expand All @@ -26,13 +32,11 @@
case reflect.Struct:
dataMap := structToMap(data[0])
for key, value := range dataMap {
shared[key] = value

Check failure on line 35 in view.go

View workflow job for this annotation

GitHub Actions / lint / nilaway

error: Potential nil panic detected. Observed nil flow from source to dereference point:
}

return &HtmlResponse{shared, receive.instance, view}
case reflect.Map:
fillShared(data[0], shared)

return &HtmlResponse{data[0], receive.instance, view}
default:
panic(fmt.Sprintf("make %s view failed, data must be map or struct", view))
Expand All @@ -42,8 +46,10 @@

func (receive *View) First(views []string, data ...any) contractshttp.Response {
for _, view := range views {
if ViewFacade.Exists(view) {
if ViewFacade != nil && ViewFacade.Exists(view) {
return receive.Make(view, data...)
} else {
fmt.Println("ViewFacade is nil or view does not exist")
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ditto

}

Expand All @@ -66,6 +72,7 @@
}
dbColumn := modelType.Field(i).Name
if modelValue.Field(i).Kind() == reflect.Pointer {
// Если поле - указатель, проверяем, не является ли оно nil
hwbrzzl marked this conversation as resolved.
Show resolved Hide resolved
if modelValue.Field(i).IsNil() {
res[dbColumn] = nil
} else {
Expand All @@ -82,6 +89,7 @@
func fillShared(data any, shared map[string]any) {
dataValue := reflect.ValueOf(data)
keys := dataValue.MapKeys()

for key, value := range shared {
exist := false
for _, k := range keys {
Expand All @@ -90,6 +98,7 @@
break
}
}

if !exist {
dataValue.SetMapIndex(reflect.ValueOf(key), reflect.ValueOf(value))
}
Expand Down
Loading