gf/util/gvalid/gvalid_check_struct.go

173 lines
5.1 KiB
Go
Raw Normal View History

// Copyright 2017-2018 gf Author(https://github.com/gogf/gf). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
package gvalid
import (
2019-06-21 22:23:07 +08:00
"strings"
2019-07-29 21:01:19 +08:00
"github.com/gogf/gf/internal/structs"
"github.com/gogf/gf/util/gconv"
)
var (
// 同时支持gvalid、valid和v标签优先使用gvalid
structTagPriority = []string{"gvalid", "valid", "v"}
)
// 校验struct对象属性object参数也可以是一个指向对象的指针返回值同CheckMap方法。
// struct的数据校验结果信息是顺序的。
2019-06-19 09:06:52 +08:00
func CheckStruct(object interface{}, rules interface{}, msgs ...CustomMsg) *Error {
// 字段别名记录用于msgs覆盖struct tag的
fieldAliases := make(map[string]string)
2019-06-19 09:06:52 +08:00
params := make(map[string]interface{})
checkRules := make(map[string]string)
customMsgs := make(CustomMsg)
// 返回的顺序规则
errorRules := make([]string, 0)
// 返回的校验错误
errorMaps := make(ErrorMap)
// 解析rules参数
switch v := rules.(type) {
// 支持校验错误顺序: []sequence tag
case []string:
for _, tag := range v {
name, rule, msg := parseSequenceTag(tag)
if len(name) == 0 {
continue
}
// 错误提示
if len(msg) > 0 {
ruleArray := strings.Split(rule, "|")
msgArray := strings.Split(msg, "|")
for k, v := range ruleArray {
// 如果msg条数比rule少那么多余的rule使用默认的错误信息
if len(msgArray) <= k {
continue
}
if len(msgArray[k]) == 0 {
continue
}
array := strings.Split(v, ":")
if _, ok := customMsgs[name]; !ok {
customMsgs[name] = make(map[string]string)
}
customMsgs[name].(map[string]string)[strings.TrimSpace(array[0])] = strings.TrimSpace(msgArray[k])
}
}
checkRules[name] = rule
errorRules = append(errorRules, name+"@"+rule)
}
// 不支持校验错误顺序: map[键名]校验规则
2019-06-19 09:06:52 +08:00
case map[string]string:
checkRules = v
}
// 首先, 按照属性循环一遍将struct的属性、数值、tag解析
for nameOrTag, field := range structs.MapField(object, structTagPriority, true) {
2019-06-19 09:06:52 +08:00
fieldName := field.Name()
params[fieldName] = field.Value()
// MapField返回map[name/tag]*Field当nameOrTag != fieldName时nameOrTag为tag
if nameOrTag != fieldName {
2019-07-13 11:47:20 +08:00
// sequence tag == struct tag, 这里的name为别名
name, rule, msg := parseSequenceTag(nameOrTag)
2019-07-13 11:47:20 +08:00
if len(name) == 0 {
name = fieldName
} else {
fieldAliases[fieldName] = name
2019-06-19 09:06:52 +08:00
}
2019-07-13 11:47:20 +08:00
// params参数使用别名**扩容**(而不仅仅使用别名),仅用于验证使用
if _, ok := params[name]; !ok {
params[name] = field.Value()
}
// 校验规则
if _, ok := checkRules[name]; !ok {
if _, ok := checkRules[fieldName]; ok {
// tag中存在别名且rules传入的参数中使用了属性命名时进行规则替换并删除该属性的规则
2019-07-13 11:47:20 +08:00
checkRules[name] = checkRules[fieldName]
delete(checkRules, fieldName)
} else {
checkRules[name] = rule
2019-06-19 09:06:52 +08:00
}
2019-07-13 11:47:20 +08:00
errorRules = append(errorRules, name+"@"+rule)
} else {
// 传递的rules规则会覆盖struct tag的规则
continue
}
// 错误提示
if len(msg) > 0 {
ruleArray := strings.Split(rule, "|")
msgArray := strings.Split(msg, "|")
for k, v := range ruleArray {
// 如果msg条数比rule少那么多余的rule使用默认的错误信息
if len(msgArray) <= k {
continue
}
if len(msgArray[k]) == 0 {
continue
}
array := strings.Split(v, ":")
if _, ok := customMsgs[name]; !ok {
customMsgs[name] = make(map[string]string)
}
customMsgs[name].(map[string]string)[strings.TrimSpace(array[0])] = strings.TrimSpace(msgArray[k])
}
2019-06-19 09:06:52 +08:00
}
}
}
2019-06-19 09:06:52 +08:00
// 自定义错误消息非必须参数优先级比rules参数中以及struct tag中定义的错误消息更高
if len(msgs) > 0 && len(msgs[0]) > 0 {
for k, v := range msgs[0] {
if a, ok := fieldAliases[k]; ok {
// 属性的别名存在时,覆盖别名的错误信息
customMsgs[a] = v
} else {
customMsgs[k] = v
2019-06-19 09:06:52 +08:00
}
}
}
2019-06-19 09:06:52 +08:00
/* 以下逻辑和CheckMap相同 */
2019-06-19 09:06:52 +08:00
// 开始执行校验: 以校验规则作为基础进行遍历校验
2019-06-21 22:23:07 +08:00
var value interface{}
2019-06-19 09:06:52 +08:00
// 这里的rule变量为多条校验规则不包含名字或者错误信息定义
for key, rule := range checkRules {
value = nil
if v, ok := params[key]; ok {
value = v
}
if e := Check(value, rule, customMsgs[key], params); e != nil {
_, item := e.FirstItem()
// 如果值为nil|""并且不需要require*验证时,其他验证失效
if value == nil || gconv.String(value) == "" {
required := false
// rule => error
2019-06-21 22:23:07 +08:00
for k := range item {
2019-06-19 09:06:52 +08:00
if _, ok := mustCheckRulesEvenValueEmpty[k]; ok {
required = true
break
}
}
if !required {
continue
}
}
if _, ok := errorMaps[key]; !ok {
errorMaps[key] = make(map[string]string)
}
for k, v := range item {
errorMaps[key][k] = v
}
}
}
if len(errorMaps) > 0 {
return newError(errorRules, errorMaps)
}
return nil
}