validator庫參數校驗若干實用技巧
本文介紹了使用validator庫做參數校驗的一些十分實用的使用技巧,包括翻譯校驗錯誤提示信息、自定義提示信息的字段名稱、自定義校驗方法等。
validator庫參數校驗若干實用技巧
在web開發中一個不可避免的環節就是對請求參數進行校驗,通常我們會在代碼中定義與請求參數相對應的模型(結構體),借助模型綁定快捷地解析請求中的參數,例如 gin 框架中的Bind
和ShouldBind
系列方法。本文就以 gin 框架的請求參數校驗為例,介紹一些validator
庫的實用技巧。
gin框架使用github.com/go-playground/validator進行參數校驗,目前已經支持github.com/go-playground/validator/v10
了,我們需要在定義結構體時使用 binding
tag標識相關校驗規則,可以查看validator文檔查看支持的所有 tag。
基本示例
首先來看gin框架內置使用validator
做參數校驗的基本示例。
package main
import (
"net/http"
"github.com/gin-gonic/gin"
)
type SignUpParam struct {
Age uint8 `json:"age" binding:"gte=1,lte=130"`
Name string `json:"name" binding:"required"`
Email string `json:"email" binding:"required,email"`
Password string `json:"password" binding:"required"`
RePassword string `json:"re_password" binding:"required,eqfield=Password"`
}
func main() {
r := gin.Default()
r.POST("/signup", func(c *gin.Context) {
var u SignUpParam
if err := c.ShouldBind(&u); err != nil {
c.JSON(http.StatusOK, gin.H{
"msg": err.Error(),
})
return
}
// 保存入庫等業務邏輯代碼...
c.JSON(http.StatusOK, "success")
})
_ = r.Run(":8999")
}
我們使用curl發送一個POST請求測試下:
curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com"}' http://127.0.0.1:8999/signup
輸出結果:
{"msg":"Key: 'SignUpParam.Email' Error:Field validation for 'Email' failed on the 'email' tag\nKey: 'SignUpParam.Password' Error:Field validation for 'Password' failed on the 'required' tag\nKey: 'SignUpParam.RePassword' Error:Field validation for 'RePassword' failed on the 'required' tag"}
從最終的輸出結果可以看到 validator
的檢驗生效了,但是錯誤提示的字段不是特別友好,我們可能需要將它翻譯成中文。
翻譯校驗錯誤提示信息
validator
庫本身是支持國際化的,借助相應的語言包可以實現校驗錯誤提示信息的自動翻譯。下面的示例代碼演示了如何將錯誤提示信息翻譯成中文,翻譯成其他語言的方法類似。
package main
import (
"fmt"
"net/http"
"github.com/gin-gonic/gin"
"github.com/gin-gonic/gin/binding"
"github.com/go-playground/locales/en"
"github.com/go-playground/locales/zh"
ut "github.com/go-playground/universal-translator"
"github.com/go-playground/validator/v10"
enTranslations "github.com/go-playground/validator/v10/translations/en"
zhTranslations "github.com/go-playground/validator/v10/translations/zh"
)
// 定義一個全局翻譯器T
var trans ut.Translator
// InitTrans 初始化翻譯器
func InitTrans(locale string) (err error) {
// 修改gin框架中的Validator引擎屬性,實現自定制
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
zhT := zh.New() // 中文翻譯器
enT := en.New() // 英文翻譯器
// 第一個參數是備用(fallback)的語言環境
// 后面的參數是應該支持的語言環境(支持多個)
// uni := ut.New(zhT, zhT) 也是可以的
uni := ut.New(enT, zhT, enT)
// locale 通常取決於 http 請求頭的 'Accept-Language'
var ok bool
// 也可以使用 uni.FindTranslator(...) 傳入多個locale進行查找
trans, ok = uni.GetTranslator(locale)
if !ok {
return fmt.Errorf("uni.GetTranslator(%s) failed", locale)
}
// 注冊翻譯器
switch locale {
case "en":
err = enTranslations.RegisterDefaultTranslations(v, trans)
case "zh":
err = zhTranslations.RegisterDefaultTranslations(v, trans)
default:
err = enTranslations.RegisterDefaultTranslations(v, trans)
}
return
}
return
}
type SignUpParam struct {
Age uint8 `json:"age" binding:"gte=1,lte=130"`
Name string `json:"name" binding:"required"`
Email string `json:"email" binding:"required,email"`
Password string `json:"password" binding:"required"`
RePassword string `json:"re_password" binding:"required,eqfield=Password"`
}
func main() {
if err := InitTrans("zh"); err != nil {
fmt.Printf("init trans failed, err:%v\n", err)
return
}
r := gin.Default()
r.POST("/signup", func(c *gin.Context) {
var u SignUpParam
if err := c.ShouldBind(&u); err != nil {
// 獲取validator.ValidationErrors類型的errors
errs, ok := err.(validator.ValidationErrors)
if !ok {
// 非validator.ValidationErrors類型錯誤直接返回
c.JSON(http.StatusOK, gin.H{
"msg": err.Error(),
})
return
}
// validator.ValidationErrors類型錯誤則進行翻譯
c.JSON(http.StatusOK, gin.H{
"msg":errs.Translate(trans),
})
return
}
// 保存入庫等具體業務邏輯代碼...
c.JSON(http.StatusOK, "success")
})
_ = r.Run(":8999")
}
同樣的請求再來一次:
curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com"}' http://127.0.0.1:8999/signup
這一次的輸出結果如下:
{"msg":{"SignUpParam.Email":"Email必須是一個有效的郵箱","SignUpParam.Password":"Password為必填字段","SignUpParam.RePassword":"RePassword為必填字段"}}
自定義錯誤提示信息的字段名
上面的錯誤提示看起來是可以了,但是還是差點意思,首先是錯誤提示中的字段並不是請求中使用的字段,例如:RePassword
是我們后端定義的結構體中的字段名,而請求中使用的是re_password
字段。如何是錯誤提示中的字段使用自定義的名稱,例如json
tag指定的值呢?
只需要在初始化翻譯器的時候像下面一樣添加一個獲取json
tag的自定義方法即可。
// InitTrans 初始化翻譯器
func InitTrans(locale string) (err error) {
// 修改gin框架中的Validator引擎屬性,實現自定制
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
// 注冊一個獲取json tag的自定義方法
v.RegisterTagNameFunc(func(fld reflect.StructField) string {
name := strings.SplitN(fld.Tag.Get("json"), ",", 2)[0]
if name == "-" {
return ""
}
return name
})
zhT := zh.New() // 中文翻譯器
enT := en.New() // 英文翻譯器
// 第一個參數是備用(fallback)的語言環境
// 后面的參數是應該支持的語言環境(支持多個)
// uni := ut.New(zhT, zhT) 也是可以的
uni := ut.New(enT, zhT, enT)
// ... liwenzhou.com ...
}
再嘗試發請求,看一下效果:
{"msg":{"SignUpParam.email":"email必須是一個有效的郵箱","SignUpParam.password":"password為必填字段","SignUpParam.re_password":"re_password為必填字段"}}
可以看到現在錯誤提示信息中使用的就是我們結構體中json
tag設置的名稱了。
但是還是有點瑕疵,那就是最終的錯誤提示信息中心還是有我們后端定義的結構體名稱——SignUpParam
,這個名稱其實是不需要隨錯誤提示返回給前端的,前端並不需要這個值。我們需要想辦法把它去掉。
這里參考https://github.com/go-playground/validator/issues/633#issuecomment-654382345提供的方法,定義一個去掉結構體名稱前綴的自定義方法:
func removeTopStruct(fields map[string]string) map[string]string {
res := map[string]string{}
for field, err := range fields {
res[field[strings.Index(field, ".")+1:]] = err
}
return res
}
我們在代碼中使用上述函數將翻譯后的errors
做一下處理即可:
if err := c.ShouldBind(&u); err != nil {
// 獲取validator.ValidationErrors類型的errors
errs, ok := err.(validator.ValidationErrors)
if !ok {
// 非validator.ValidationErrors類型錯誤直接返回
c.JSON(http.StatusOK, gin.H{
"msg": err.Error(),
})
return
}
// validator.ValidationErrors類型錯誤則進行翻譯
// 並使用removeTopStruct函數去除字段名中的結構體名稱標識
c.JSON(http.StatusOK, gin.H{
"msg": removeTopStruct(errs.Translate(trans)),
})
return
}
看一下最終的效果:
{"msg":{"email":"email必須是一個有效的郵箱","password":"password為必填字段","re_password":"re_password為必填字段"}}
這一次看起來就比較符合我們預期的標准了。
自定義結構體校驗方法
上面的校驗還是有點小問題,就是當涉及到一些復雜的校驗規則,比如re_password
字段需要與password
字段的值相等這樣的校驗規則,我們的自定義錯誤提示字段名稱方法就不能很好解決錯誤提示信息中的其他字段名稱了。
curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com","password":"123","re_password":"321"}' http://127.0.0.1:8999/signup
最后輸出的錯誤提示信息如下:
{"msg":{"email":"email必須是一個有效的郵箱","re_password":"re_password必須等於Password"}}
可以看到re_password
字段的提示信息中還是出現了Password
這個結構體字段名稱。這有點小小的遺憾,畢竟自定義字段名稱的方法不能影響被當成param傳入的值。
此時如果想要追求更好的提示效果,將上面的Password字段也改為和json
tag一致的名稱,就需要我們自定義結構體校驗的方法。
例如,我們為SignUpParam
自定義一個校驗方法如下:
// SignUpParamStructLevelValidation 自定義SignUpParam結構體校驗函數
func SignUpParamStructLevelValidation(sl validator.StructLevel) {
su := sl.Current().Interface().(SignUpParam)
if su.Password != su.RePassword {
// 輸出錯誤提示信息,最后一個參數就是傳遞的param
sl.ReportError(su.RePassword, "re_password", "RePassword", "eqfield", "password")
}
}
然后在初始化校驗器的函數中注冊該自定義校驗方法即可:
func InitTrans(locale string) (err error) {
// 修改gin框架中的Validator引擎屬性,實現自定制
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
// ... liwenzhou.com ...
// 為SignUpParam注冊自定義校驗方法
v.RegisterStructValidation(SignUpParamStructLevelValidation, SignUpParam{})
zhT := zh.New() // 中文翻譯器
enT := en.New() // 英文翻譯器
// ... liwenzhou.com ...
}
最終再請求一次,看一下效果:
{"msg":{"email":"email必須是一個有效的郵箱","re_password":"re_password必須等於password"}}
這一次re_password
字段的錯誤提示信息就符合我們預期了。
自定義字段校驗方法
除了上面介紹到的自定義結構體校驗方法,validator
還支持為某個字段自定義校驗方法,並使用RegisterValidation()
注冊到校驗器實例中。
接下來我們來為SignUpParam
添加一個需要使用自定義校驗方法checkDate
做參數校驗的字段Date
。
type SignUpParam struct {
Age uint8 `json:"age" binding:"gte=1,lte=130"`
Name string `json:"name" binding:"required"`
Email string `json:"email" binding:"required,email"`
Password string `json:"password" binding:"required"`
RePassword string `json:"re_password" binding:"required,eqfield=Password"`
// 需要使用自定義校驗方法checkDate做參數校驗的字段Date
Date string `json:"date" binding:"required,datetime=2006-01-02,checkDate"`
}
其中datetime=2006-01-02
是內置的用於校驗日期類參數是否滿足指定格式要求的tag。 如果傳入的date
參數不滿足2006-01-02
這種格式就會提示如下錯誤:
{"msg":{"date":"date的格式必須是2006-01-02"}}
針對date字段除了內置的datetime=2006-01-02
提供的格式要求外,假設我們還要求該字段的時間必須是一個未來的時間(晚於當前時間),像這樣針對某個字段的特殊校驗需求就需要我們使用自定義字段校驗方法了。
首先我們要在需要執行自定義校驗的字段后面添加自定義tag,這里使用的是checkDate
,注意使用英文分號分隔開。
// customFunc 自定義字段級別校驗方法
func customFunc(fl validator.FieldLevel) bool {
date, err := time.Parse("2006-01-02", fl.Field().String())
if err != nil {
return false
}
if date.Before(time.Now()) {
return false
}
return true
}
定義好了字段及其自定義校驗方法后,就需要將它們聯系起來並注冊到我們的校驗器實例中。
// 在校驗器注冊自定義的校驗方法
if err := v.RegisterValidation("checkDate", customFunc); err != nil {
return err
}
這樣,我們就可以對請求參數中date
字段執行自定義的checkDate
進行校驗了。 我們發送如下請求測試一下:
curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123@qq.com","password":"123", "re_password": "123", "date":"2020-01-02"}' http://127.0.0.1:8999/signup
此時得到的響應結果是:
{"msg":{"date":"Key: 'SignUpParam.date' Error:Field validation for 'date' failed on the 'checkDate' tag"}}
這…自定義字段級別的校驗方法的錯誤提示信息很“簡單粗暴”,和我們上面的中文提示風格有出入,必須想辦法搞定它呀!
自定義翻譯方法
我們現在需要為自定義字段校驗方法提供一個自定義的翻譯方法,從而實現該字段錯誤提示信息的自定義顯示。
// registerTranslator 為自定義字段添加翻譯功能
func registerTranslator(tag string, msg string) validator.RegisterTranslationsFunc {
return func(trans ut.Translator) error {
if err := trans.Add(tag, msg, false); err != nil {
return err
}
return nil
}
}
// translate 自定義字段的翻譯方法
func translate(trans ut.Translator, fe validator.FieldError) string {
msg, err := trans.T(fe.Tag(), fe.Field())
if err != nil {
panic(fe.(error).Error())
}
return msg
}
定義好了相關翻譯方法之后,我們在InitTrans
函數中通過調用RegisterTranslation()
方法來注冊我們自定義的翻譯方法。
// InitTrans 初始化翻譯器
func InitTrans(locale string) (err error) {
// ...liwenzhou.com...
// 注冊翻譯器
switch locale {
case "en":
err = enTranslations.RegisterDefaultTranslations(v, trans)
case "zh":
err = zhTranslations.RegisterDefaultTranslations(v, trans)
default:
err = enTranslations.RegisterDefaultTranslations(v, trans)
}
if err != nil {
return err
}
// 注意!因為這里會使用到trans實例
// 所以這一步注冊要放到trans初始化的后面
if err := v.RegisterTranslation(
"checkDate",
trans,
registerTranslator("checkDate", "{0}必須要晚於當前日期"),
translate,
); err != nil {
return err
}
return
}
return
}
這樣再次嘗試發送請求,就能得到想要的錯誤提示信息了。
{"msg":{"date":"date必須要晚於當前日期"}}
總結
本文總結的gin框架中validator
的使用技巧同樣也適用於直接使用validator
庫,區別僅僅在於我們配置的是gin框架中的校驗器還是由validator.New()
創建的校驗器。同時使用validator
庫確實能夠在一定程度上減少我們的編碼量,但是它不太可能完美解決我們所有需求,所以你需要找到兩者之間的平衡點。
參考鏈接:
https://github.com/go-playground/validator/blob/master/_examples/simple/main.go
https://github.com/go-playground/validator/blob/master/_examples/translations/main.go
https://github.com/go-playground/validator/issues/567
https://github.com/go-playground/validator/issues/633
https://github.com/go-playground/validator/issues/551