Compare commits

...

10 commits

Author SHA1 Message Date
eea72ac9bc 🔀 Merge branch 'rel/0.2.0' into stable 2024-02-07 09:19:14 -06:00
c42d835d97 📝 Update CHANGELOG 2024-02-07 09:18:44 -06:00
7edba56f11 ⬆️ Upgrade responsable-errors 2024-02-07 09:15:04 -06:00
30765808e2 🔀 Merge branch 'bugfix/abort-after-err' into develop 2024-02-07 08:50:20 -06:00
119d73aae1 🐛 Only send response if not already sent 2024-02-07 08:49:09 -06:00
644fbe5fa5 🐛 Abort after error
Fixes https://codeberg.org/danjones000/gin-error-handler/issues/1
2024-02-07 08:41:54 -06:00
8e8a8079c5 🧪 Add integration test 2024-02-07 08:40:26 -06:00
0d8b624c4f 🔀 Merge tag 'v0.1.0' into develop
🔖🎉 Initial release
2024-01-22 22:21:41 -06:00
d46826c43e 🔀 Merge branch 'rel/0.1.0' into stable 2024-01-22 22:21:00 -06:00
097419f5ef 📝 Better documentation.
Also a Changelog.
2024-01-22 22:20:04 -06:00
7 changed files with 163 additions and 7 deletions

24
CHANGELOG.md Normal file
View file

@ -0,0 +1,24 @@
# Changelog
## [0.2.0] - 2024-02-07
### Fixed
- [Using HandlerWithError as Middleware allows response to be sent](https://codeberg.org/danjones000/gin-error-handler/issues/1)
- Error response still sent if other response is already sent, resulting in duplicate body
## [0.1.0] - 2024-01-22
🎉 Initial release
### Added
- HandlerWithError
+ HandlerWithErrorWrapper
- ErrorMiddleware
+ Option
* Transformer
- WithTransformer
- WithDefaultTransformer
* LoggerFunc
- WithLogger

View file

@ -1,3 +1,87 @@
# Gin Error Handler # Gin Error Handler
A Gin middleware and wrapper functions to make handling errors easier. A Gin middleware and wrapper functions to make handling errors easier.
## Installation
Use the module in the usual way: `codeberg.org/danjones000/gin-error-handler`.
## Usage
```go
package main
import (
"net/http"
handler "codeberg.org/danjones000/gin-error-handler"
rErrors "codeberg.org/danjones000/responsable-errors"
"github.com/go-playground/validator/v10"
"github.com/gin-gonic/gin"
)
var _ rErrors.ResponsableError = new(vError)
type vError struct {
validator.ValidationErrors
}
func (ve *vError) Msg() string {
return "Validation Error"
}
func (ve *vError) Status() int {
return http.StatusBadRequest
}
func (ve *vError) JSON() any {
errs := make([]map[string]string, len(ve))
for i, fe := range ve {
errs[i] = map[string]string{"field":fe.Field(),"error":fe.Error()}
}
return map[string][]map[string]string{"errors":errs}
}
func (ve *vError) MarshalJSON() ([]byte, error) {
return json.Marshal(e.JSON())
}
func (ve *vError) Unwrap() error {
return ve.ValidationErrors
}
func main() {
r := gin.New()
r.Use(handler.ErrorMiddleware(
handler.WithTransformer(func (err error) rErrors.ResponsableError {
var ve validator.ValidationErrors
if !errors.As(err, &ve) {
return nil
}
return &vError{ve}
}),
handler.WithDefaultTransformer(),
handler.WithLogger(ctx context.Context, err rErrors.ResponsableError) {
log.Print(err)
})
r.GET("/user", handler.HandlerWithErrorWrapper(func (c *gin.Context) error {
var qu struct {
id int `binding:"required"`
}{}
if err := c.ShouldBindQuery(&qu); err != nil {
return err
}
user, err := user.Get(qu.id)
if err != nil {
return err
}
if user == nil {
return rErrors.NewNotFound("User not found")
}
c.JSON(200, user)
}))
}
```
In our example, if `ShouldBindQuery` fails validation, we'll return a nicely formatted error with each validation error, due to our custom transformer. If no user is found, we'll return a 404, with an error message. And any errors are logged to stdout.

2
go.mod
View file

@ -3,7 +3,7 @@ module codeberg.org/danjones000/gin-error-handler
go 1.21.5 go 1.21.5
require ( require (
codeberg.org/danjones000/responsable-errors v0.2.0 codeberg.org/danjones000/responsable-errors v0.2.1
github.com/gin-gonic/gin v1.9.1 github.com/gin-gonic/gin v1.9.1
github.com/stretchr/testify v1.8.4 github.com/stretchr/testify v1.8.4
) )

6
go.sum
View file

@ -1,7 +1,5 @@
codeberg.org/danjones000/responsable-errors v0.1.1 h1:WTWo0egPNsp+kEKRK8R+yc/bfdEDUbwEjMnqgA7IklA= codeberg.org/danjones000/responsable-errors v0.2.1 h1:Ur6t+QZWaAfBgjQWBqSCgS3sTySdkKQD/FQefeRrm90=
codeberg.org/danjones000/responsable-errors v0.1.1/go.mod h1:susEj39A/bflyej4tRirtuVKkmfUdhR2Skljwd/1ndI= codeberg.org/danjones000/responsable-errors v0.2.1/go.mod h1:susEj39A/bflyej4tRirtuVKkmfUdhR2Skljwd/1ndI=
codeberg.org/danjones000/responsable-errors v0.2.0 h1:WUTBSaKQzkXHDYraHt3mxB4QHqVk78sKR/VeteXM1OY=
codeberg.org/danjones000/responsable-errors v0.2.0/go.mod h1:susEj39A/bflyej4tRirtuVKkmfUdhR2Skljwd/1ndI=
github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM=
github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s=
github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=

View file

@ -25,6 +25,8 @@ func HandlerWithErrorWrapper(h HandlerWithError) gin.HandlerFunc {
return func(c *gin.Context) { return func(c *gin.Context) {
err := h(c) err := h(c)
if err != nil { if err != nil {
// We shouldn't process more handlers if there was an error
c.Abort()
c.Error(err) c.Error(err)
} }
} }

47
int_test.go Normal file
View file

@ -0,0 +1,47 @@
package handler
import (
"net/http"
"net/http/httptest"
"testing"
errors "codeberg.org/danjones000/responsable-errors"
"github.com/gin-gonic/gin"
"github.com/stretchr/testify/assert"
)
func TestErrorMiddleware(t *testing.T) {
gin.SetMode(gin.TestMode)
w := httptest.NewRecorder()
_, r := gin.CreateTestContext(w)
r.Use(ErrorMiddleware())
r.Use(HandlerWithErrorWrapper(func(c *gin.Context) error {
return errors.Errorf(400, "Oops")
}))
r.GET("/", func(c *gin.Context) {
c.JSON(200, gin.H{"Hello": "World"})
})
req, _ := http.NewRequest("GET", "/", nil)
r.ServeHTTP(w, req)
assert.Equal(t, 400, w.Code)
assert.Equal(t, `{"error":"Oops"}`, w.Body.String())
}
func TestErrorNoResponseIfAlreadyWritten(t *testing.T) {
gin.SetMode(gin.TestMode)
w := httptest.NewRecorder()
_, r := gin.CreateTestContext(w)
r.Use(ErrorMiddleware())
r.GET("/", HandlerWithErrorWrapper(func(c *gin.Context) error {
c.JSON(200, gin.H{"Hello": "World"})
return errors.Errorf(400, "Oops")
}))
req, _ := http.NewRequest("GET", "/", nil)
r.ServeHTTP(w, req)
assert.Equal(t, 200, w.Code)
assert.Equal(t, `{"Hello":"World"}`, w.Body.String())
}

View file

@ -55,7 +55,8 @@ func ErrorMiddleware(opts ...Option) gin.HandlerFunc {
c.Set("rendered_error", re) c.Set("rendered_error", re)
// @todo check a response hasn't already been sent if !c.Writer.Written() {
c.JSON(re.Status(), re) c.JSON(re.Status(), re)
}
} }
} }