mirror of
https://github.com/astaxie/beego.git
synced 2024-11-25 16:30:54 +00:00
233 lines
4.8 KiB
Go
233 lines
4.8 KiB
Go
package validation
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
VALIDTAG = "valid"
|
|
)
|
|
|
|
var (
|
|
// key: function name
|
|
// value: the number of parameters
|
|
funcs = make(Funcs)
|
|
|
|
// doesn't belong to validation functions
|
|
unFuncs = map[string]bool{
|
|
"Clear": true,
|
|
"HasErrors": true,
|
|
"ErrorMap": true,
|
|
"Error": true,
|
|
"apply": true,
|
|
"Check": true,
|
|
"Valid": true,
|
|
"NoMatch": true,
|
|
}
|
|
)
|
|
|
|
func init() {
|
|
v := &Validation{}
|
|
t := reflect.TypeOf(v)
|
|
for i := 0; i < t.NumMethod(); i++ {
|
|
m := t.Method(i)
|
|
if !unFuncs[m.Name] {
|
|
funcs[m.Name] = m.Func
|
|
}
|
|
}
|
|
}
|
|
|
|
// Valid function type
|
|
type ValidFunc struct {
|
|
Name string
|
|
Params []interface{}
|
|
}
|
|
|
|
// Validate function map
|
|
type Funcs map[string]reflect.Value
|
|
|
|
// validate values with named type string
|
|
func (f Funcs) Call(name string, params ...interface{}) (result []reflect.Value, err error) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
err = fmt.Errorf("%v", r)
|
|
}
|
|
}()
|
|
if _, ok := f[name]; !ok {
|
|
err = fmt.Errorf("%s does not exist", name)
|
|
return
|
|
}
|
|
if len(params) != f[name].Type().NumIn() {
|
|
err = fmt.Errorf("The number of params is not adapted")
|
|
return
|
|
}
|
|
in := make([]reflect.Value, len(params))
|
|
for k, param := range params {
|
|
in[k] = reflect.ValueOf(param)
|
|
}
|
|
result = f[name].Call(in)
|
|
return
|
|
}
|
|
|
|
func isStruct(t reflect.Type) bool {
|
|
return t.Kind() == reflect.Struct
|
|
}
|
|
|
|
func isStructPtr(t reflect.Type) bool {
|
|
return t.Kind() == reflect.Ptr && t.Elem().Kind() == reflect.Struct
|
|
}
|
|
|
|
func getValidFuncs(f reflect.StructField) (vfs []ValidFunc, err error) {
|
|
tag := f.Tag.Get(VALIDTAG)
|
|
if len(tag) == 0 {
|
|
return
|
|
}
|
|
if vfs, tag, err = getRegFuncs(tag, f.Name); err != nil {
|
|
fmt.Printf("%+v\n", err)
|
|
return
|
|
}
|
|
fs := strings.Split(tag, ";")
|
|
for _, vfunc := range fs {
|
|
var vf ValidFunc
|
|
if len(vfunc) == 0 {
|
|
continue
|
|
}
|
|
vf, err = parseFunc(vfunc, f.Name)
|
|
if err != nil {
|
|
return
|
|
}
|
|
vfs = append(vfs, vf)
|
|
}
|
|
return
|
|
}
|
|
|
|
// Get Match function
|
|
// May be get NoMatch function in the future
|
|
func getRegFuncs(tag, key string) (vfs []ValidFunc, str string, err error) {
|
|
tag = strings.TrimSpace(tag)
|
|
index := strings.Index(tag, "Match(/")
|
|
if index == -1 {
|
|
str = tag
|
|
return
|
|
}
|
|
end := strings.LastIndex(tag, "/)")
|
|
if end < index {
|
|
err = fmt.Errorf("invalid Match function")
|
|
return
|
|
}
|
|
reg, err := regexp.Compile(tag[index+len("Match(/") : end])
|
|
if err != nil {
|
|
return
|
|
}
|
|
vfs = []ValidFunc{ValidFunc{"Match", []interface{}{reg, key + ".Match"}}}
|
|
str = strings.TrimSpace(tag[:index]) + strings.TrimSpace(tag[end+len("/)"):])
|
|
return
|
|
}
|
|
|
|
func parseFunc(vfunc, key string) (v ValidFunc, err error) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
err = fmt.Errorf("%v", r)
|
|
}
|
|
}()
|
|
|
|
vfunc = strings.TrimSpace(vfunc)
|
|
start := strings.Index(vfunc, "(")
|
|
var num int
|
|
|
|
// doesn't need parameter valid function
|
|
if start == -1 {
|
|
if num, err = numIn(vfunc); err != nil {
|
|
return
|
|
}
|
|
if num != 0 {
|
|
err = fmt.Errorf("%s require %d parameters", vfunc, num)
|
|
return
|
|
}
|
|
v = ValidFunc{vfunc, []interface{}{key + "." + vfunc}}
|
|
return
|
|
}
|
|
|
|
end := strings.Index(vfunc, ")")
|
|
if end == -1 {
|
|
err = fmt.Errorf("invalid valid function")
|
|
return
|
|
}
|
|
|
|
name := strings.TrimSpace(vfunc[:start])
|
|
if num, err = numIn(name); err != nil {
|
|
return
|
|
}
|
|
|
|
params := strings.Split(vfunc[start+1:end], ",")
|
|
// the num of param must be equal
|
|
if num != len(params) {
|
|
err = fmt.Errorf("%s require %d parameters", name, num)
|
|
return
|
|
}
|
|
|
|
tParams, err := trim(name, key+"."+name, params)
|
|
if err != nil {
|
|
return
|
|
}
|
|
v = ValidFunc{name, tParams}
|
|
return
|
|
}
|
|
|
|
func numIn(name string) (num int, err error) {
|
|
fn, ok := funcs[name]
|
|
if !ok {
|
|
err = fmt.Errorf("doesn't exsits %s valid function", name)
|
|
return
|
|
}
|
|
// sub *Validation obj and key
|
|
num = fn.Type().NumIn() - 3
|
|
return
|
|
}
|
|
|
|
func trim(name, key string, s []string) (ts []interface{}, err error) {
|
|
ts = make([]interface{}, len(s), len(s)+1)
|
|
fn, ok := funcs[name]
|
|
if !ok {
|
|
err = fmt.Errorf("doesn't exsits %s valid function", name)
|
|
return
|
|
}
|
|
for i := 0; i < len(s); i++ {
|
|
var param interface{}
|
|
// skip *Validation and obj params
|
|
if param, err = magic(fn.Type().In(i+2), strings.TrimSpace(s[i])); err != nil {
|
|
return
|
|
}
|
|
ts[i] = param
|
|
}
|
|
ts = append(ts, key)
|
|
return
|
|
}
|
|
|
|
// modify the parameters's type to adapt the function input parameters' type
|
|
func magic(t reflect.Type, s string) (i interface{}, err error) {
|
|
switch t.Kind() {
|
|
case reflect.Int:
|
|
i, err = strconv.Atoi(s)
|
|
case reflect.String:
|
|
i = s
|
|
case reflect.Ptr:
|
|
if t.Elem().String() != "regexp.Regexp" {
|
|
err = fmt.Errorf("does not support %s", t.Elem().String())
|
|
return
|
|
}
|
|
i, err = regexp.Compile(s)
|
|
default:
|
|
err = fmt.Errorf("does not support %s", t.Kind().String())
|
|
}
|
|
return
|
|
}
|
|
|
|
func mergeParam(v *Validation, obj interface{}, params []interface{}) []interface{} {
|
|
return append([]interface{}{v, obj}, params...)
|
|
}
|