From 6ecdfc8a242ef841e3de71ac5a2ce119ef2d5ccf Mon Sep 17 00:00:00 2001 From: toimtoimtoim Date: Sun, 13 Dec 2020 01:04:20 +0200 Subject: [PATCH] Improve request binding documentation. See Echo pull request #1681 (https://github.com/labstack/echo/pull/1681) --- website/content/guide/request.md | 33 +++++++++++++++++++++++++++++++- 1 file changed, 32 insertions(+), 1 deletion(-) diff --git a/website/content/guide/request.md b/website/content/guide/request.md index 48e77d4e..6ab37049 100644 --- a/website/content/guide/request.md +++ b/website/content/guide/request.md @@ -8,10 +8,32 @@ description = "Handling HTTP request in Echo" ## Bind Data -To bind request body into a Go type use `Context#Bind(i interface{})`. +Echo provides following method to bind data from different sources (route params, query params, request body) to structure +`Context#Bind(i interface{})` method. The default binder supports decoding application/json, application/xml and application/x-www-form-urlencoded data based on the Content-Type header. +Request data is binded to the struct in given order: + +1. Route parameters +2. Query parameters +3. Request body + +Notes: + +* Each step can overwrite binded fields from the previous step. This means if your json request has query param + `&name=query` and body is `{"name": "body"}` then the result will be `User{Name: "body"}`. +* To avoid security flaws try to avoid passing binded structs directly to other methods if + these structs contain fields that should not be bindable. It is advisable to have separate struct for binding and map it + explicitly to your business struct. Consider what will happen if your binded struct has public + field `IsAdmin bool` and request body would contain `{IsAdmin: true, Name: "hacker"}`. +* To bind data only from request body use following code + ```go + if err := (&DefaultBinder{}).BindBody(c, &payload); err != nil { + return err + } + ``` + Example below binds the request payload into `User` struct based on tags: ```go @@ -29,6 +51,15 @@ func(c echo.Context) (err error) { if err = c.Bind(u); err != nil { return } + // To avoid security flaws try to avoid passing binded structs directly to other methods + // if these structs contain fields that should not be bindable. + user := UserDTO{ + Name: u.Name, + Email: u.Email, + IsAdmin: false // because you could accidentally expose fields that should not be bind + } + executeSomeBusinessLogic(user) + return c.JSON(http.StatusOK, u) } ```