go-zero中使用validtor库进行参数校验
本文介绍了使用validator库做参数校验的一些十分实用的使用技巧,包括翻译校验错误提示信息、自定义提示信息的字段名称、自定义校验方法等。
validator库参数校验若干实用技巧
在web开发中一个不可避免的环节就是对请求参数进行校验,通常我们会在代码中定义与请求参数相对应的模型(结构体),借助模型绑定快捷地解析请求中的参数。本文就以 go-zero 框架的请求参数校验为例,介绍一些validator
库的实用技巧。
go-zero框架使用github.com/go-playgrou…进行参数校验,目前已经支持github.com/go-playground/validator/v10
了,我们需要在定义结构体时使用 validate
tag标识相关校验规则,可以查看validator文档查看支持的所有 tag。
安装validator库
go get github.com/go-playground/validator/v10
基本示例
首先来看go-zero框架内置使用validator
做参数校验的基本示例。
在api层编写Req时进行validator标签的编辑
生成.go文件的时候type就会带上相关标签(types里面)
type (
TestReq {
Age int64 `json:"age" validate:"gte=1,lte=130"`
Name string `json:"name" validate:"required"`
Email string `json:"email" validate:"required,email"`
Password string `json:"password" validate:"required"`
RePassword string `json:"re_password" validate:"required,eqfield=Password"`
}
TestResp {
}
)
在handler层调用validator库
这个是没有使用validator前的handler
func TestApiHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
var req types.TestReq
if err := httpx.Parse(r, &req); err != nil {
httpx.ErrorCtx(r.Context(), w, err)
return
}
l := lottery.NewTestApiLogic(r.Context(), svcCtx)
resp, err := l.TestApi(&req)
if err != nil {
result.ParamErrorResult(r, w, err)
} else {
result.HttpResult(r, w, resp, err)
}
}
}
使用validator校验
err := validator.New().StructCtx(r.Context(), req)
if err != nil {
result.ParamErrorResult(r, w, err)
return
}
后续配合模板,直接生成这段代码即可,这样handler层可以不必再编辑
翻译校验错误提示信息
validator
库本身是支持国际化的,借助相应的语言包可以实现校验错误提示信息的自动翻译。下面的示例代码演示了如何将错误提示信息翻译成中文,翻译成其他语言的方法类似。
安装翻译相关包
go get github.com/go-playground/universal-translator
编写validate的方法
func Validate(dataStruct interface{}) error {
zh_ch := zh.New()
validate := validator.New()
// 注册一个函数,获取struct tag里自定义的label作为字段名
validate.RegisterTagNameFunc(func(fld reflect.StructField) string {
name := strings.SplitN(fld.Tag.Get("json"), ",", 2)[0]
if name == "-" {
return ""
}
return name
})
uni := ut.New(zh_ch)
trans, _ := uni.GetTranslator("zh")
// 验证器注册翻译器
zh_translations.RegisterDefaultTranslations(validate, trans)
err := validate.Struct(dataStruct)
if err != nil {
for _, err := range err.(validator.ValidationErrors) {
return errors.New(err.Translate(trans))
}
}
return nil
}
编写handler中的拦截(直接放模板里面去)
//err := validator.New().StructCtx(r.Context(), req)
//if err != nil {
// httpx.ErrorCtx(r.Context(), w, err)
// return
//}
// 上面validator.New()出来的校验实例注释掉,用自己实现的带翻译器的校验
validateErr := Validate(&req)
if validateErr != nil {
result.ParamErrorResult(r, w, validateErr)
return
}
自定义结构体校验方法
上面的校验还是有点小问题,就是当涉及到一些复杂的校验规则,比如re_password
字段需要与password
字段的值相等这样的校验规则,我们的自定义错误提示字段名称方法就不能很好解决错误提示信息中的其他字段名称了。
当我们测试re_password字段的时候,可以看到re_password
字段的提示信息中还是出现了Password
这个结构体字段名称。这有点小小的遗憾,毕竟自定义字段名称的方法不能影响被当成param传入的值。
{
"code": 100002,
"msg": "参数错误 ,re_password必须等于Password"
}
此时如果想要追求更好的提示效果,将上面的Password字段也改为和json
tag一致的名称,就需要我们自定义结构体校验的方法。
例如,我们为SignUpParam
自定义一个校验方法如下:(可以把它写在handler下)
// SignUpParamStructLevelValidation 自定义SignUpParam结构体校验函数
func SignUpParamStructLevelValidation(sl validator.StructLevel) {
su := sl.Current().Interface().(types.TestReq)
if su.Password != su.RePassword {
// 输出错误提示信息,最后一个参数就是传递的param
sl.ReportError(su.RePassword, "re_password", "RePassword", "eqfield", "password")
}
}
并且删掉原先的校验
eqfield=Password // api中的这个校验不要,重新生成.go文件
然后在初始化校验器的函数Validate中注册该自定义校验方法即可:
// 注册自定义结构体校验方法
validate.RegisterStructValidation(SignUpParamStructLevelValidation, types.TestReq{})
最终再请求一次,看一下效果:
{
"code": 100002,
"msg": "参数错误 ,re_password必须等于password"
}
这一次re_password
字段的错误提示信息就符合我们预期了。
自定义字段校验方法
除了上面介绍到的自定义结构体校验方法,validator
还支持为某个字段自定义校验方法,并使用RegisterValidation()
注册到校验器实例中。
接下来我们来为SignUpParam
添加一个需要使用自定义校验方法checkDate
做参数校验的字段Date
。
TestReq {
Age int64 `json:"age" validate:"gte=1,lte=130"`
Name string `json:"name" validate:"required"`
Email string `json:"email" validate:"required,email"`
Password string `json:"password" validate:"required"`
RePassword string `json:"re_password" validate:"required"`
// 需要使用自定义校验方法checkDate做参数校验的字段Date
Date string `json:"date" validate:"required,datetime=2006-01-02,checkDate"`
}
其中datetime=2006-01-02
是内置的用于校验日期类参数是否满足指定格式要求的tag。 如果传入的date
参数不满足2006-01-02
这种格式就会提示如下错误:
{
"code": 100002,
"msg": "参数错误 ,date的格式必须是2006-01-02"
}
针对date字段除了内置的datetime=2006-01-02
提供的格式要求外,假设我们还要求该字段的时间必须是一个未来的时间(晚于当前时间),像这样针对某个字段的特殊校验需求就需要我们使用自定义字段校验方法了。
首先我们要在需要执行自定义校验的字段后面添加自定义tag,这里使用的是checkDate
,注意使用英文逗号分隔开。(写在handler下)
// checkDate 自定义字段级别校验方法
func checkDate(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
}
定义好了字段及其自定义校验方法后,就需要将它们联系起来并注册到我们的校验器实例中。(在Validate函数)
// 注册自定义结构体字段校验方法
if err := validate.RegisterValidation("checkDate", checkDate); err != nil {
return err
}
这样,我们就可以对请求参数中date
字段执行自定义的checkDate
进行校验了。 我们发送如下请求测试一下,得到响应结果如下:
{
"code": 100002,
"msg": "参数错误 ,Key: 'TestReq.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()
方法来注册我们自定义的翻译方法。(Validate函数下)
// InitTrans 初始化翻译器
func Validate(dataStruct interface{}) error {
zh_ch := zh.New()
validate := validator.New()
// 注册一个函数,获取struct tag里自定义的label作为字段名
validate.RegisterTagNameFunc(func(fld reflect.StructField) string {
name := strings.SplitN(fld.Tag.Get("json"), ",", 2)[0]
if name == "-" {
return ""
}
return name
})
// 在这里注册自定义结构体/字段校验方法
uni := ut.New(zh_ch)
trans, _ := uni.GetTranslator("zh")
// 注意!因为这里会使用到trans实例
// 所以这一步注册要放到trans初始化的后面
if err := validate.RegisterTranslation(
"checkDate",
trans,
registerTranslator("checkDate", "{0}必须要晚于当前日期"),
translate,
); err != nil {
return err
}
return
}
return
}
这样再次尝试发送请求,就能得到想要的错误提示信息了。
{
"code": 100002,
"msg": "参数错误 ,date的格式必须是2006-01-02"
}
总结
本文总结的go-zero框架中validator
的使用技巧同样也适用于直接使用validator
库,区别仅仅在于我们配置的是go-zero框架中的校验器还是由validator.New()
创建的校验器。同时使用validator
库确实能够在一定程度上减少我们的编码量,但是它不太可能完美解决我们所有需求,所以你需要找到两者之间的平衡点。
答疑
- 如果需要定制化中文翻译,可以将zh相关包copy下来进行定制化
- 目前是单独有一个validator目录,存放翻译器以及注册自定义校验,避免循环导包;也避免放在全局,因为internal的相关代码是不对外暴露的。
参考链接
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通