mirror of
https://gitee.com/johng/gf.git
synced 2024-11-30 03:07:45 +08:00
165 lines
4.3 KiB
Go
165 lines
4.3 KiB
Go
// 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 (
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/gogf/gf/v2/internal/empty"
|
|
"github.com/gogf/gf/v2/util/gconv"
|
|
"github.com/gogf/gf/v2/util/gutil"
|
|
)
|
|
|
|
type checkRequiredInput struct {
|
|
Value interface{} // Value to be validated.
|
|
RuleKey string // RuleKey is like the "max" in rule "max: 6"
|
|
RulePattern string // RulePattern is like "6" in rule:"max:6"
|
|
DataMap map[string]interface{} // Parameter map.
|
|
CaseInsensitive bool // Case-Insensitive comparison.
|
|
}
|
|
|
|
// checkRequired checks `value` using required rules.
|
|
// It also supports require checks for `value` of type: slice, map.
|
|
func (v *Validator) checkRequired(in checkRequiredInput) bool {
|
|
required := false
|
|
switch in.RuleKey {
|
|
// Required.
|
|
case "required":
|
|
required = true
|
|
|
|
// Required unless all given field and its value are equal.
|
|
// Example: required-if: id,1,age,18
|
|
case "required-if":
|
|
required = false
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
// It supports multiple field and value pairs.
|
|
if len(array)%2 == 0 {
|
|
for i := 0; i < len(array); {
|
|
tk := array[i]
|
|
tv := array[i+1]
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, tk)
|
|
if in.CaseInsensitive {
|
|
required = strings.EqualFold(tv, gconv.String(foundValue))
|
|
} else {
|
|
required = strings.Compare(tv, gconv.String(foundValue)) == 0
|
|
}
|
|
if required {
|
|
break
|
|
}
|
|
i += 2
|
|
}
|
|
}
|
|
|
|
// Required unless all given field and its value are not equal.
|
|
// Example: required-unless: id,1,age,18
|
|
case "required-unless":
|
|
required = true
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
// It supports multiple field and value pairs.
|
|
if len(array)%2 == 0 {
|
|
for i := 0; i < len(array); {
|
|
tk := array[i]
|
|
tv := array[i+1]
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, tk)
|
|
if in.CaseInsensitive {
|
|
required = !strings.EqualFold(tv, gconv.String(foundValue))
|
|
} else {
|
|
required = strings.Compare(tv, gconv.String(foundValue)) != 0
|
|
}
|
|
if !required {
|
|
break
|
|
}
|
|
i += 2
|
|
}
|
|
}
|
|
|
|
// Required if any of given fields are not empty.
|
|
// Example: required-with:id,name
|
|
case "required-with":
|
|
required = false
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
for i := 0; i < len(array); i++ {
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, array[i])
|
|
if !empty.IsEmpty(foundValue) {
|
|
required = true
|
|
break
|
|
}
|
|
}
|
|
|
|
// Required if all given fields are not empty.
|
|
// Example: required-with:id,name
|
|
case "required-with-all":
|
|
required = true
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
for i := 0; i < len(array); i++ {
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, array[i])
|
|
if empty.IsEmpty(foundValue) {
|
|
required = false
|
|
break
|
|
}
|
|
}
|
|
|
|
// Required if any of given fields are empty.
|
|
// Example: required-with:id,name
|
|
case "required-without":
|
|
required = false
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
for i := 0; i < len(array); i++ {
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, array[i])
|
|
if empty.IsEmpty(foundValue) {
|
|
required = true
|
|
break
|
|
}
|
|
}
|
|
|
|
// Required if all given fields are empty.
|
|
// Example: required-with:id,name
|
|
case "required-without-all":
|
|
required = true
|
|
var (
|
|
array = strings.Split(in.RulePattern, ",")
|
|
foundValue interface{}
|
|
)
|
|
for i := 0; i < len(array); i++ {
|
|
_, foundValue = gutil.MapPossibleItemByKey(in.DataMap, array[i])
|
|
if !empty.IsEmpty(foundValue) {
|
|
required = false
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if required {
|
|
reflectValue := reflect.ValueOf(in.Value)
|
|
for reflectValue.Kind() == reflect.Ptr {
|
|
reflectValue = reflectValue.Elem()
|
|
}
|
|
switch reflectValue.Kind() {
|
|
case reflect.String, reflect.Map, reflect.Array, reflect.Slice:
|
|
return reflectValue.Len() != 0
|
|
}
|
|
return gconv.String(in.Value) != ""
|
|
} else {
|
|
return true
|
|
}
|
|
}
|