2021-01-12 10:46:39 +08:00
|
|
|
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
2018-11-13 00:12:35 +08:00
|
|
|
//
|
|
|
|
// 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,
|
2019-02-02 16:18:25 +08:00
|
|
|
// You can obtain one at https://github.com/gogf/gf.
|
2018-11-13 00:12:35 +08:00
|
|
|
|
|
|
|
package gvalid
|
|
|
|
|
|
|
|
import (
|
2021-10-11 21:34:19 +08:00
|
|
|
"context"
|
2021-11-09 17:50:14 +08:00
|
|
|
"errors"
|
2021-11-22 20:28:47 +08:00
|
|
|
"reflect"
|
2021-11-13 23:30:31 +08:00
|
|
|
"strings"
|
|
|
|
|
2021-10-11 21:41:56 +08:00
|
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
2022-03-11 10:24:42 +08:00
|
|
|
"github.com/gogf/gf/v2/internal/reflection"
|
2021-10-11 21:41:56 +08:00
|
|
|
"github.com/gogf/gf/v2/util/gconv"
|
2018-11-13 00:12:35 +08:00
|
|
|
)
|
|
|
|
|
2021-10-11 21:34:19 +08:00
|
|
|
func (v *Validator) doCheckMap(ctx context.Context, params interface{}) Error {
|
2021-11-22 20:28:47 +08:00
|
|
|
if params == 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.
|
2021-11-09 17:50:14 +08:00
|
|
|
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 {
|
2022-03-02 20:00:40 +08:00
|
|
|
name, rule, msg := ParseTagValue(tag)
|
2019-06-19 09:06:52 +08:00
|
|
|
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
|
|
|
}
|
2018-11-13 00:12:35 +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
|
|
|
}
|
2021-11-22 20:28:47 +08:00
|
|
|
inputParamMap := gconv.Map(params)
|
|
|
|
if inputParamMap == nil {
|
2021-11-09 17:50:14 +08:00
|
|
|
return newValidationErrorByStr(
|
2021-05-19 13:29:40 +08:00
|
|
|
internalParamsErrRuleName,
|
2021-11-09 17:50:14 +08:00
|
|
|
errors.New("invalid params type: convert to map failed"),
|
2020-12-09 21:00:30 +08:00
|
|
|
)
|
|
|
|
}
|
2021-05-19 09:25:49 +08:00
|
|
|
if msg, ok := v.messages.(CustomMsg); ok && len(msg) > 0 {
|
2021-08-01 23:50:44 +08:00
|
|
|
if len(customMessage) > 0 {
|
2021-05-19 09:25:49 +08:00
|
|
|
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 (
|
2021-11-22 20:28:47 +08:00
|
|
|
value interface{}
|
|
|
|
validator = v.Clone()
|
2021-08-01 23:50:44 +08:00
|
|
|
)
|
2021-11-22 20:28:47 +08:00
|
|
|
|
|
|
|
// It checks the struct recursively if its attribute is an embedded struct.
|
2022-07-15 10:49:04 +08:00
|
|
|
// Ignore inputParamMap, assoc, rules and messages from parent.
|
|
|
|
validator.assoc = nil
|
2021-11-22 20:28:47 +08:00
|
|
|
validator.rules = nil
|
|
|
|
validator.messages = nil
|
|
|
|
for _, item := range inputParamMap {
|
2022-03-11 10:24:42 +08:00
|
|
|
originTypeAndKind := reflection.OriginTypeAndKind(item)
|
2021-11-22 20:28:47 +08:00
|
|
|
switch originTypeAndKind.OriginKind {
|
|
|
|
case reflect.Map, reflect.Struct, reflect.Slice, reflect.Array:
|
|
|
|
v.doCheckValueRecursively(ctx, doCheckValueRecursivelyInput{
|
2022-05-26 15:28:17 +08:00
|
|
|
Value: item,
|
|
|
|
Type: originTypeAndKind.InputType,
|
|
|
|
Kind: originTypeAndKind.OriginKind,
|
|
|
|
ErrorMaps: errorMaps,
|
2021-11-22 20:28:47 +08:00
|
|
|
})
|
|
|
|
}
|
|
|
|
// Bail feature.
|
|
|
|
if v.bail && len(errorMaps) > 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if v.bail && len(errorMaps) > 0 {
|
|
|
|
return newValidationError(gcode.CodeValidationFailed, nil, errorMaps)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The following logic is the same as some of CheckStruct but without sequence support.
|
2021-08-01 23:50:44 +08:00
|
|
|
for _, checkRuleItem := range checkRules {
|
|
|
|
if len(checkRuleItem.Rule) == 0 {
|
2019-06-19 09:06:52 +08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
value = nil
|
2021-11-22 20:28:47 +08:00
|
|
|
if valueItem, ok := inputParamMap[checkRuleItem.Name]; ok {
|
2021-08-01 23:50:44 +08:00
|
|
|
value = valueItem
|
2019-06-19 09:06:52 +08:00
|
|
|
}
|
2020-10-21 00:08:36 +08:00
|
|
|
// It checks each rule and its value in loop.
|
2021-10-11 21:34:19 +08:00
|
|
|
if validatedError := v.doCheckValue(ctx, doCheckValueInput{
|
2023-10-09 20:00:08 +08:00
|
|
|
Name: checkRuleItem.Name,
|
|
|
|
Value: value,
|
|
|
|
ValueType: reflect.TypeOf(value),
|
|
|
|
Rule: checkRuleItem.Rule,
|
|
|
|
Messages: customMessage[checkRuleItem.Name],
|
|
|
|
DataRaw: params,
|
|
|
|
DataMap: inputParamMap,
|
2021-08-01 23:50:44 +08:00
|
|
|
}); validatedError != nil {
|
|
|
|
_, errorItem := validatedError.FirstItem()
|
2020-08-20 23:25:36 +08:00
|
|
|
// ===========================================================
|
2021-11-14 21:00:34 +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
|
|
|
// ===========================================================
|
2020-06-14 17:28:48 +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 {
|
2022-09-26 22:11:13 +08:00
|
|
|
if required = v.checkRuleRequired(ruleKey); required {
|
2019-06-19 09:06:52 +08:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !required {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2021-08-01 23:50:44 +08:00
|
|
|
if _, ok := errorMaps[checkRuleItem.Name]; !ok {
|
2021-11-09 17:50:14 +08:00
|
|
|
errorMaps[checkRuleItem.Name] = make(map[string]error)
|
2021-08-01 23:50:44 +08:00
|
|
|
}
|
2021-11-14 21:00:34 +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 {
|
2021-11-09 17:50:14 +08:00
|
|
|
return newValidationError(gcode.CodeValidationFailed, checkRules, errorMaps)
|
2019-06-19 09:06:52 +08:00
|
|
|
}
|
|
|
|
return nil
|
2018-11-13 00:12:35 +08:00
|
|
|
}
|