gin/binding/default_validator.go

96 lines
2.3 KiB
Go
Raw Normal View History

// Copyright 2017 Manu Martinez-Almeida. All rights reserved.
2017-06-12 09:04:52 +03:00
// Use of this source code is governed by a MIT style
// license that can be found in the LICENSE file.
2015-05-31 17:18:50 +03:00
package binding
import (
"reflect"
"strconv"
"strings"
2015-05-31 17:18:50 +03:00
"sync"
"github.com/go-playground/validator/v10"
2015-05-31 17:18:50 +03:00
)
type defaultValidator struct {
once sync.Once
validate *validator.Validate
}
type SliceValidationError []error
// Error concatenates all error elements in SliceValidationError into a single string separated by \n.
func (err SliceValidationError) Error() string {
if len(err) == 0 {
return ""
}
var b strings.Builder
for i := 0; i < len(err); i++ {
if err[i] != nil {
if b.Len() > 0 {
b.WriteString("\n")
}
b.WriteString("[" + strconv.Itoa(i) + "]: " + err[i].Error())
}
}
return b.String()
}
var _ StructValidator = (*defaultValidator)(nil)
2015-05-31 17:18:50 +03:00
// ValidateStruct receives any kind of type, but only performed struct or pointer to struct type.
2022-03-21 04:43:17 +03:00
func (v *defaultValidator) ValidateStruct(obj any) error {
if obj == nil {
return nil
}
value := reflect.ValueOf(obj)
switch value.Kind() {
case reflect.Ptr:
if value.Elem().Kind() != reflect.Struct {
return v.ValidateStruct(value.Elem().Interface())
}
return v.validateStruct(obj)
case reflect.Struct:
return v.validateStruct(obj)
case reflect.Slice, reflect.Array:
count := value.Len()
validateRet := make(SliceValidationError, 0)
for i := 0; i < count; i++ {
if err := v.ValidateStruct(value.Index(i).Interface()); err != nil {
validateRet = append(validateRet, err)
}
}
if len(validateRet) == 0 {
return nil
2015-05-31 17:30:00 +03:00
}
return validateRet
default:
return nil
2015-05-31 17:18:50 +03:00
}
}
// validateStruct receives struct type
2022-03-21 04:43:17 +03:00
func (v *defaultValidator) validateStruct(obj any) error {
v.lazyinit()
return v.validate.Struct(obj)
2015-05-31 17:18:50 +03:00
}
// Engine returns the underlying validator engine which powers the default
// Validator instance. This is useful if you want to register custom validations
// or struct level validations. See validator GoDoc for more info -
// https://pkg.go.dev/github.com/go-playground/validator/v10
2022-03-21 04:43:17 +03:00
func (v *defaultValidator) Engine() any {
v.lazyinit()
return v.validate
}
2015-05-31 17:18:50 +03:00
func (v *defaultValidator) lazyinit() {
v.once.Do(func() {
v.validate = validator.New()
v.validate.SetTagName("binding")
2015-05-31 17:18:50 +03:00
})
}