gf/util/gvalid/gvalid_validator_check_map.go

155 lines
4.1 KiB
Go
Raw Normal View History

2021-01-12 10:46:39 +08:00
// Copyright GoFrame Author(https://goframe.org). 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 (
"context"
"errors"
2021-11-13 23:30:31 +08:00
"strings"
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/errors/gcode"
"github.com/gogf/gf/v2/util/gconv"
)
2020-05-10 17:49:23 +08:00
// CheckMap validates map and returns the error result. It returns nil if with successful validation.
// The parameter `params` should be type of map.
func (v *Validator) CheckMap(ctx context.Context, params interface{}) Error {
return v.doCheckMap(ctx, params)
}
func (v *Validator) doCheckMap(ctx context.Context, params interface{}) Error {
2020-12-09 21:00:30 +08:00
// If there's no validation rules, it does nothing and returns quickly.
if params == nil || v.rules == nil {
2020-12-09 21:00:30 +08:00
return nil
2019-06-19 09:06:52 +08:00
}
2020-05-10 16:48:00 +08:00
var (
2021-08-01 23:50:44 +08:00
checkRules = make([]fieldRule, 0)
customMessage = make(CustomMsg) // map[RuleKey]ErrorMsg.
errorMaps = make(map[string]map[string]error)
2020-05-10 16:48:00 +08:00
)
2021-08-01 23:50:44 +08:00
switch assertValue := v.rules.(type) {
2020-05-10 17:49:23 +08:00
// Sequence tag: []sequence tag
// Sequence has order for error results.
2019-06-19 09:06:52 +08:00
case []string:
2021-08-01 23:50:44 +08:00
for _, tag := range assertValue {
2019-06-19 09:06:52 +08:00
name, rule, msg := parseSequenceTag(tag)
if len(name) == 0 {
continue
}
if len(msg) > 0 {
2020-05-10 17:49:23 +08:00
var (
msgArray = strings.Split(msg, "|")
ruleArray = strings.Split(rule, "|")
)
2021-08-01 23:50:44 +08:00
for k, ruleItem := range ruleArray {
2020-05-10 17:49:23 +08:00
// If length of custom messages is lesser than length of rules,
// the rest rules use the default error messages.
2019-06-19 09:06:52 +08:00
if len(msgArray) <= k {
continue
}
if len(msgArray[k]) == 0 {
continue
}
2021-08-01 23:50:44 +08:00
array := strings.Split(ruleItem, ":")
if _, ok := customMessage[name]; !ok {
customMessage[name] = make(map[string]string)
2019-06-19 09:06:52 +08:00
}
2021-08-01 23:50:44 +08:00
customMessage[name].(map[string]string)[strings.TrimSpace(array[0])] = strings.TrimSpace(msgArray[k])
2019-06-19 09:06:52 +08:00
}
}
2021-08-01 23:50:44 +08:00
checkRules = append(checkRules, fieldRule{
Name: name,
Rule: rule,
})
2019-06-19 09:06:52 +08:00
}
2020-05-10 17:49:23 +08:00
// No sequence rules: map[field]rule
2019-06-19 09:06:52 +08:00
case map[string]string:
2021-08-01 23:50:44 +08:00
for name, rule := range assertValue {
checkRules = append(checkRules, fieldRule{
Name: name,
Rule: rule,
})
}
2019-06-19 09:06:52 +08:00
}
2020-12-09 21:00:30 +08:00
// If there's no validation rules, it does nothing and returns quickly.
if len(checkRules) == 0 {
return nil
}
data := gconv.Map(params)
if data == nil {
return newValidationErrorByStr(
2021-05-19 13:29:40 +08:00
internalParamsErrRuleName,
errors.New("invalid params type: convert to map failed"),
2020-12-09 21:00:30 +08:00
)
}
if msg, ok := v.messages.(CustomMsg); ok && len(msg) > 0 {
2021-08-01 23:50:44 +08:00
if len(customMessage) > 0 {
for k, v := range msg {
2021-08-01 23:50:44 +08:00
customMessage[k] = v
2019-06-19 09:06:52 +08:00
}
} else {
2021-08-01 23:50:44 +08:00
customMessage = msg
2019-06-19 09:06:52 +08:00
}
}
2021-08-01 23:50:44 +08:00
var (
value interface{}
)
for _, checkRuleItem := range checkRules {
if len(checkRuleItem.Rule) == 0 {
2019-06-19 09:06:52 +08:00
continue
}
value = nil
2021-08-01 23:50:44 +08:00
if valueItem, ok := data[checkRuleItem.Name]; ok {
value = valueItem
2019-06-19 09:06:52 +08:00
}
// It checks each rule and its value in loop.
if validatedError := v.doCheckValue(ctx, doCheckValueInput{
2021-08-01 23:50:44 +08:00
Name: checkRuleItem.Name,
Value: value,
Rule: checkRuleItem.Rule,
Messages: customMessage[checkRuleItem.Name],
DataRaw: params,
DataMap: data,
}); validatedError != nil {
_, errorItem := validatedError.FirstItem()
2020-08-20 23:25:36 +08:00
// ===========================================================
// Only in map and struct validations:
// If value is nil or empty string and has no required* rules,
// it clears the error message.
2020-08-20 23:25:36 +08:00
// ===========================================================
if gconv.String(value) == "" {
2019-06-19 09:06:52 +08:00
required := false
// rule => error
2021-08-01 23:50:44 +08:00
for ruleKey := range errorItem {
// Default required rules.
2021-08-01 23:50:44 +08:00
if _, ok := mustCheckRulesEvenValueEmpty[ruleKey]; ok {
2019-06-19 09:06:52 +08:00
required = true
break
}
}
if !required {
continue
}
}
2021-08-01 23:50:44 +08:00
if _, ok := errorMaps[checkRuleItem.Name]; !ok {
errorMaps[checkRuleItem.Name] = make(map[string]error)
2021-08-01 23:50:44 +08:00
}
for ruleKey, ruleError := range errorItem {
errorMaps[checkRuleItem.Name][ruleKey] = ruleError
2019-06-19 09:06:52 +08:00
}
2021-08-01 23:50:44 +08:00
if v.bail {
break
2019-06-19 09:06:52 +08:00
}
}
}
if len(errorMaps) > 0 {
return newValidationError(gcode.CodeValidationFailed, checkRules, errorMaps)
2019-06-19 09:06:52 +08:00
}
return nil
}