Vinculação de Modelo e Validação
Para vincular uma corpo de requisição à um tipo, use a vinculação de modelo. Nós atualmente suportamos a vinculação de JSON, XML, YAML e valores de formulário padrão (foo=bar&boo=baz).
A Gin usa go-playground/validator/v10
para a validação. Consulte a documentação completa sobre o uso de marcadores nesta ligação.
Nota que precisas de definir o marcador da vinculação correspondente em todos os campos que quiseres vincular. Por exemplo, quando estiveres a vincular a partir de JSON, defina json:"fieldname"
.
Além disto, a Gin fornece dois conjuntos de métodos para vinculação:
- Tipo - Deve vincular
- Métodos -
Bind
,BindJSON
,BindXML
,BindQuery
,BindYAML
- Comportamento - Estes métodos usam
MustBindWith
nos bastidores. Se houver um erro de vinculação, a requisição é abortada comc.AbortWithError(400, err).SetType(ErrorTypeBind)
. Isto define o código do estado da resposta para 400 e o cabeçalhoContent-Type
é definido paratext/plain; charset=utf-8
. Nota que se tentares definir o código da resposta depois disto, resultará em um aviso[GIN-debug] [WARNING] Headers were already written. Wanted to override status code 400 with 422
. Se desejas ter maior controlo sobre o comportamento, considere usar o método equivalenteShouldBind
.
- Métodos -
- Tipo - Deveria vincular
- Métodos -
ShouldBind
,ShouldBindJSON
,ShouldBindXML
,ShouldBindQuery
,ShouldBindYAML
- Comportamento - Estes métodos usam
ShouldBindWith
nos bastidores. Se houver um erro de vinculação, o erro é retornado e é responsabilidade do programador manipular a requisição e o erro apropriadamente.
- Métodos -
Quando estiveres o método Bind
, a Gin tenta inferir o vinculador dependendo do cabeçalho do Content-Type
. Se estiveres certo daquilo que estiveres a vincular, podes usar MustBindWith
ou ShouldBindWith
.
Tu podes também especificar que os campos específicos são obrigatório. Se um campo for decorado com binding:"required"
e tiver um valor vazio quando estiveres a vincular, um erro será retornado:
// Vinculando a partir de JSONtype Login struct { User string `form:"user" json:"user" xml:"user" binding:"required"` Password string `form:"password" json:"password" xml:"password" binding:"required"`}
func main() { router := gin.Default()
// Exemplo para vincular o JSON ({"user": "manu", "password": "123"}) router.POST("/loginJSON", func(c *gin.Context) { var json Login if err := c.ShouldBindJSON(&json); err != nil { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) return }
if json.User != "manu" || json.Password != "123" { c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"}) return }
c.JSON(http.StatusOK, gin.H{"status": "you are logged in"}) })
// Exemplo pra vincular o XML ( // <?xml version="1.0" encoding="UTF-8"?> // <root> // <user>manu</user> // <password>123</password> // </root>) router.POST("/loginXML", func(c *gin.Context) { var xml Login if err := c.ShouldBindXML(&xml); err != nil { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) return }
if xml.User != "manu" || xml.Password != "123" { c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"}) return }
c.JSON(http.StatusOK, gin.H{"status": "you are logged in"}) })
// Exemplo para vincular um formulário de HTML (user=manu&password=123) router.POST("/loginForm", func(c *gin.Context) { var form Login // Isto inferirá qual vinculador usar dependendo do cabeçalho do `content-type`. if err := c.ShouldBind(&form); err != nil { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) return }
if form.User != "manu" || form.Password != "123" { c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"}) return }
c.JSON(http.StatusOK, gin.H{"status": "you are logged in"}) })
// Ouvir e servir na 0.0.0.0:8080 router.Run(":8080")}
Requisição Simples
$ curl -v -X POST \ http://localhost:8080/loginJSON \ -H 'content-type: application/json' \ -d '{ "user": "manu" }'> POST /loginJSON HTTP/1.1> Host: localhost:8080> User-Agent: curl/7.51.0> Accept: */*> content-type: application/json> Content-Length: 18>* upload completely sent off: 18 out of 18 bytes< HTTP/1.1 400 Bad Request< Content-Type: application/json; charset=utf-8< Date: Fri, 04 Aug 2017 03:51:31 GMT< Content-Length: 100<{"error":"Key: 'Login.Password' Error:Field validation for 'Password' failed on the 'required' tag"}
Saltar a Validação
Quando estiveres a executar o exemplo de cima usando o comando curl
de cima, isto retorna erro. Porque o exemplo usa binding:"required"
para Password
. Se usares binding:"-"
para Password
, então este não retornará erro quando estiveres a executar o exemplo de cima novamente.