• 模型绑定和验证
    • 示例请求
    • 忽略验证

    模型绑定和验证

    要将请求体绑定到结构体中,使用模型绑定。 Gin目前支持JSON、XML、YAML和标准表单值的绑定(foo=bar&boo=baz)。

    Gin使用 go-playground/validator.v8 进行验证。 查看标签用法的全部文档.

    使用时,需要在要绑定的所有字段上,设置相应的tag。 例如,使用 JSON 绑定时,设置字段标签为 json:"fieldname"

    Gin提供了两类绑定方法:

    • Type - Must bind
      • Methods - Bind, BindJSON, BindXML, BindQuery, BindYAML
      • Behavior - 这些方法属于 MustBindWith 的具体调用。 如果发生绑定错误,则请求终止,并触发 c.AbortWithError(400, err).SetType(ErrorTypeBind)。响应状态码被设置为 400 并且 Content-Type 被设置为 text/plain; charset=utf-8。 如果您在此之后尝试设置响应状态码,Gin会输出日志 [GIN-debug] [WARNING] Headers were already written. Wanted to override status code 400 with 422。 如果您希望更好地控制绑定,考虑使用 ShouldBind 等效方法。
    • Type - Should bind
      • Methods - ShouldBind, ShouldBindJSON, ShouldBindXML, ShouldBindQuery, ShouldBindYAML
      • Behavior - 这些方法属于 ShouldBindWith 的具体调用。 如果发生绑定错误,Gin 会返回错误并由开发者处理错误和请求。使用 Bind 方法时,Gin 会尝试根据 Content-Type 推断如何绑定。 如果你明确知道要绑定什么,可以使用 MustBindWithShouldBindWith

    你也可以指定必须绑定的字段。 如果一个字段的 tag 加上了 binding:"required",但绑定时是空值, Gin 会报错。

    1. // 绑定 JSON
    2. type Login struct {
    3. User string `form:"user" json:"user" xml:"user" binding:"required"`
    4. Password string `form:"password" json:"password" xml:"password" binding:"required"`
    5. }
    6. func main() {
    7. router := gin.Default()
    8. // 绑定 JSON ({"user": "manu", "password": "123"})
    9. router.POST("/loginJSON", func(c *gin.Context) {
    10. var json Login
    11. if err := c.ShouldBindJSON(&json); err != nil {
    12. c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
    13. return
    14. }
    15. if json.User != "manu" || json.Password != "123" {
    16. c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"})
    17. return
    18. }
    19. c.JSON(http.StatusOK, gin.H{"status": "you are logged in"})
    20. })
    21. // 绑定 XML (
    22. // <?xml version="1.0" encoding="UTF-8"?>
    23. // <root>
    24. // <user>user</user>
    25. // <password>123</password>
    26. // </root>)
    27. router.POST("/loginXML", func(c *gin.Context) {
    28. var xml Login
    29. if err := c.ShouldBindXML(&xml); err != nil {
    30. c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
    31. return
    32. }
    33. if xml.User != "manu" || xml.Password != "123" {
    34. c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"})
    35. return
    36. }
    37. c.JSON(http.StatusOK, gin.H{"status": "you are logged in"})
    38. })
    39. // 绑定 HTML 表单 (user=manu&password=123)
    40. router.POST("/loginForm", func(c *gin.Context) {
    41. var form Login
    42. // 根据 Content-Type Header 推断使用哪个绑定器。
    43. if err := c.ShouldBind(&form); err != nil {
    44. c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
    45. return
    46. }
    47. if form.User != "manu" || form.Password != "123" {
    48. c.JSON(http.StatusUnauthorized, gin.H{"status": "unauthorized"})
    49. return
    50. }
    51. c.JSON(http.StatusOK, gin.H{"status": "you are logged in"})
    52. })
    53. // 监听并在 0.0.0.0:8080 上启动服务
    54. router.Run(":8080")
    55. }

    示例请求

    1. $ curl -v -X POST \
    2. http://localhost:8080/loginJSON \
    3. -H 'content-type: application/json' \
    4. -d '{ "user": "manu" }'
    5. > POST /loginJSON HTTP/1.1
    6. > Host: localhost:8080
    7. > User-Agent: curl/7.51.0
    8. > Accept: */*
    9. > content-type: application/json
    10. > Content-Length: 18
    11. >
    12. * upload completely sent off: 18 out of 18 bytes
    13. < HTTP/1.1 400 Bad Request
    14. < Content-Type: application/json; charset=utf-8
    15. < Date: Fri, 04 Aug 2017 03:51:31 GMT
    16. < Content-Length: 100
    17. <
    18. {"error":"Key: 'Login.Password' Error:Field validation for 'Password' failed on the 'required' tag"}

    忽略验证

    使用上述的 curl 命令运行上面的示例时会返回错误。 因为示例中 Password 使用了 binding:"required"。 如果 Password 使用 binding:"-", 再次运行上面的示例就不会返回错误。