From 5f69a0dbdac252aba3af4e4cd33745c76c6b75bb Mon Sep 17 00:00:00 2001 From: jc <46619361+juancwu@users.noreply.github.com> Date: Thu, 14 Nov 2024 01:12:02 -0500 Subject: [PATCH 1/5] add req validator middleware --- internal/middleware/req_validator.go | 55 ++++++++++++ internal/middleware/req_validator_test.go | 103 ++++++++++++++++++++++ 2 files changed, 158 insertions(+) create mode 100644 internal/middleware/req_validator.go create mode 100644 internal/middleware/req_validator_test.go diff --git a/internal/middleware/req_validator.go b/internal/middleware/req_validator.go new file mode 100644 index 0000000..dfd8efb --- /dev/null +++ b/internal/middleware/req_validator.go @@ -0,0 +1,55 @@ +package middleware + +import ( + "net/http" + "reflect" + + "github.com/go-playground/validator/v10" + "github.com/labstack/echo/v4" + "github.com/rs/zerolog/log" +) + +// Struct solely exists to comply with Echo's interface to add a custom validator... +type RequestBodyValidator struct { + validator *validator.Validate +} + +func (rv *RequestBodyValidator) Validate(i interface{}) error { + log.Info().Msgf("Validating struct: %+v\n", i) + if err := rv.validator.Struct(i); err != nil { + log.Error().Err(err).Msg("Validation error") + return err + } + + return nil +} + +// Creates a new request validator that can be set to an Echo instance +// and used for validating request bodies with c.Validate() +func NewRequestBodyValidator() *RequestBodyValidator { + return &RequestBodyValidator{validator: validator.New()} +} + +// Middleware that validates the incoming request body with the given structType. +// Optionally pass any argument(s) like the fmt.Sprintf() method to customize +// a log message before ending the connection. +func ValidateRequestBody(structType reflect.Type, args ...interface{}) echo.MiddlewareFunc { + return func(next echo.HandlerFunc) echo.HandlerFunc { + return func(c echo.Context) error { + reqStruct := reflect.New(structType) + + if err := c.Bind(reqStruct.Interface()); err != nil { + return echo.NewHTTPError(http.StatusBadRequest, "Invalid request body.") + } + + if err := c.Validate(reqStruct.Interface()); err != nil { + // this will let the global error handler handle + // the ValidationError and get error string for + // the each invalid field. + return err + } + + return next(c) + } + } +} diff --git a/internal/middleware/req_validator_test.go b/internal/middleware/req_validator_test.go new file mode 100644 index 0000000..9f232e2 --- /dev/null +++ b/internal/middleware/req_validator_test.go @@ -0,0 +1,103 @@ +package middleware + +import ( + "bytes" + "encoding/json" + "net/http" + "net/http/httptest" + "reflect" + "testing" + + "github.com/labstack/echo/v4" + "github.com/stretchr/testify/assert" +) + +func TestRequestBodyValidator(t *testing.T) { + type testStruct struct { + TestField bool `json:"test_field" validate:"required"` + } + + e := echo.New() + e.Validator = NewRequestBodyValidator() + e.POST("/", handler, ValidateRequestBody(reflect.TypeOf(testStruct{}))) + + tests := []struct { + name string + payload interface{} + expectedCode int + }{ + { + name: "Valid request body", + payload: testStruct{ + TestField: true, + }, + expectedCode: http.StatusOK, + }, + { + name: "Invalid request body - validation error", + payload: testStruct{ + // will fail required validation + TestField: false, + }, + // expecting 500 since the middleware its expected to return + // the original ValidationErrors from validator pkg + expectedCode: http.StatusInternalServerError, + }, + { + name: "Empty request body", + payload: nil, + // expecting 500 since the middleware its expected to return + // the original ValidationErrors from validator pkg + expectedCode: http.StatusInternalServerError, + }, + { + name: "Invalid JSON format", + payload: `{ + "test_field": invalid + }`, + expectedCode: http.StatusBadRequest, + }, + { + name: "Wrong type in JSON", + payload: map[string]interface{}{ + "test_field": "not a boolean", + }, + expectedCode: http.StatusBadRequest, + }, + } + + for _, tc := range tests { + t.Run(tc.name, func(t *testing.T) { + var req *http.Request + + if tc.payload != nil { + var payload []byte + var err error + + // handle string payloads (for invalid JSON tests) + if strPayload, ok := tc.payload.(string); ok { + payload = []byte(strPayload) + } else { + payload, err = json.Marshal(tc.payload) + assert.NoError(t, err) + } + + req = httptest.NewRequest(http.MethodPost, "/", bytes.NewReader(payload)) + req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON) + } else { + req = httptest.NewRequest(http.MethodPost, "/", nil) + req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON) + } + + rec := httptest.NewRecorder() + e.ServeHTTP(rec, req) + + assert.Equal(t, tc.expectedCode, rec.Code) + }) + } +} + +// test handler +func handler(c echo.Context) error { + return c.String(http.StatusOK, "pass") +} From 717fa9bf82ee0e45246c28fd8ba67342143a3192 Mon Sep 17 00:00:00 2001 From: jc <46619361+juancwu@users.noreply.github.com> Date: Thu, 14 Nov 2024 01:13:29 -0500 Subject: [PATCH 2/5] remove server's custom validator type --- internal/server/index.go | 2 +- internal/server/types.go | 24 ------------------------ 2 files changed, 1 insertion(+), 25 deletions(-) diff --git a/internal/server/index.go b/internal/server/index.go index 22e160f..b3a8a50 100644 --- a/internal/server/index.go +++ b/internal/server/index.go @@ -69,7 +69,7 @@ func New(testing bool) (*Server, error) { e.Use(echoMiddleware.Recover()) e.Use(apiLimiter.RateLimit()) // global rate limit - customValidator := NewCustomValidator() + customValidator := middleware.NewRequestBodyValidator() fmt.Printf("Initializing validator: %+v\n", customValidator) e.Validator = customValidator diff --git a/internal/server/types.go b/internal/server/types.go index 7f516ad..fcbed28 100644 --- a/internal/server/types.go +++ b/internal/server/types.go @@ -1,12 +1,7 @@ package server import ( - "fmt" - "net/http" "time" - - "github.com/go-playground/validator/v10" - "github.com/labstack/echo/v4" ) type DatabaseInfo struct { @@ -43,25 +38,6 @@ type CreateResourceRequestRequest struct { Status string `json:"status" validate:"required"` } -type CustomValidator struct { - validator *validator.Validate -} - -func NewCustomValidator() *CustomValidator { - v := validator.New() - return &CustomValidator{validator: v} -} - -func (cv *CustomValidator) Validate(i interface{}) error { - fmt.Printf("Validating struct: %+v\n", i) - if err := cv.validator.Struct(i); err != nil { - fmt.Printf("Validation error: %v\n", err) - return echo.NewHTTPError(http.StatusBadRequest, err.Error()) - } - - return nil -} - type SignupRequest struct { Email string `json:"email" validate:"required,email"` Password string `json:"password" validate:"required,min=8"` From e42861b16441c7be723ff3f2923c14dc63b82632 Mon Sep 17 00:00:00 2001 From: jc <46619361+juancwu@users.noreply.github.com> Date: Thu, 14 Nov 2024 01:13:41 -0500 Subject: [PATCH 3/5] update validateBody method to return original ValidationErrors --- internal/server/handler_helpers.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/internal/server/handler_helpers.go b/internal/server/handler_helpers.go index 066e69e..62c13e4 100644 --- a/internal/server/handler_helpers.go +++ b/internal/server/handler_helpers.go @@ -14,7 +14,10 @@ func validateBody(c echo.Context, requestBodyType interface{}) error { } if err := c.Validate(requestBodyType); err != nil { - return echo.NewHTTPError(http.StatusBadRequest, err.Error()) + // this will let the global error handler handle + // the ValidationError and get error string for + // the each invalid field. + return err } return nil From 116223ad7cedd1f90c0b3579d99a7a790dd8ab2f Mon Sep 17 00:00:00 2001 From: jc <46619361+juancwu@users.noreply.github.com> Date: Thu, 14 Nov 2024 12:31:52 -0500 Subject: [PATCH 4/5] remove custom log msg found out that go doesn't like it when formatting directives are used when the function signature is not fmt.Printf or Namef --- internal/middleware/req_validator.go | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/internal/middleware/req_validator.go b/internal/middleware/req_validator.go index dfd8efb..7cdf442 100644 --- a/internal/middleware/req_validator.go +++ b/internal/middleware/req_validator.go @@ -31,9 +31,7 @@ func NewRequestBodyValidator() *RequestBodyValidator { } // Middleware that validates the incoming request body with the given structType. -// Optionally pass any argument(s) like the fmt.Sprintf() method to customize -// a log message before ending the connection. -func ValidateRequestBody(structType reflect.Type, args ...interface{}) echo.MiddlewareFunc { +func ValidateRequestBody(structType reflect.Type) echo.MiddlewareFunc { return func(next echo.HandlerFunc) echo.HandlerFunc { return func(c echo.Context) error { reqStruct := reflect.New(structType) From ef913145957cbfbe5dd1473aab830663b1d8998b Mon Sep 17 00:00:00 2001 From: jc <46619361+juancwu@users.noreply.github.com> Date: Fri, 15 Nov 2024 01:41:24 -0500 Subject: [PATCH 5/5] format binding error into response --- internal/middleware/req_validator.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/internal/middleware/req_validator.go b/internal/middleware/req_validator.go index 7cdf442..feef71c 100644 --- a/internal/middleware/req_validator.go +++ b/internal/middleware/req_validator.go @@ -1,6 +1,7 @@ package middleware import ( + "fmt" "net/http" "reflect" @@ -37,7 +38,7 @@ func ValidateRequestBody(structType reflect.Type) echo.MiddlewareFunc { reqStruct := reflect.New(structType) if err := c.Bind(reqStruct.Interface()); err != nil { - return echo.NewHTTPError(http.StatusBadRequest, "Invalid request body.") + return echo.NewHTTPError(http.StatusBadRequest, fmt.Sprintf("Invalid request body: %v", err)) } if err := c.Validate(reqStruct.Interface()); err != nil {