Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
446 lines
14 KiB
446 lines
14 KiB
// Copyright 2015 go-swagger maintainers |
|
// |
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
|
// you may not use this file except in compliance with the License. |
|
// You may obtain a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, |
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
// See the License for the specific language governing permissions and |
|
// limitations under the License. |
|
|
|
package validate |
|
|
|
import ( |
|
"context" |
|
"fmt" |
|
"reflect" |
|
"strings" |
|
"unicode/utf8" |
|
|
|
"github.com/go-openapi/errors" |
|
"github.com/go-openapi/strfmt" |
|
"github.com/go-openapi/swag" |
|
) |
|
|
|
// Enum validates if the data is a member of the enum |
|
func Enum(path, in string, data interface{}, enum interface{}) *errors.Validation { |
|
return EnumCase(path, in, data, enum, true) |
|
} |
|
|
|
// EnumCase validates if the data is a member of the enum and may respect case-sensitivity for strings |
|
func EnumCase(path, in string, data interface{}, enum interface{}, caseSensitive bool) *errors.Validation { |
|
val := reflect.ValueOf(enum) |
|
if val.Kind() != reflect.Slice { |
|
return nil |
|
} |
|
|
|
dataString := convertEnumCaseStringKind(data, caseSensitive) |
|
var values []interface{} |
|
for i := 0; i < val.Len(); i++ { |
|
ele := val.Index(i) |
|
enumValue := ele.Interface() |
|
if data != nil { |
|
if reflect.DeepEqual(data, enumValue) { |
|
return nil |
|
} |
|
enumString := convertEnumCaseStringKind(enumValue, caseSensitive) |
|
if dataString != nil && enumString != nil && strings.EqualFold(*dataString, *enumString) { |
|
return nil |
|
} |
|
actualType := reflect.TypeOf(enumValue) |
|
if actualType == nil { // Safeguard. Frankly, I don't know how we may get a nil |
|
continue |
|
} |
|
expectedValue := reflect.ValueOf(data) |
|
if expectedValue.IsValid() && expectedValue.Type().ConvertibleTo(actualType) { |
|
// Attempt comparison after type conversion |
|
if reflect.DeepEqual(expectedValue.Convert(actualType).Interface(), enumValue) { |
|
return nil |
|
} |
|
} |
|
} |
|
values = append(values, enumValue) |
|
} |
|
return errors.EnumFail(path, in, data, values) |
|
} |
|
|
|
// convertEnumCaseStringKind converts interface if it is kind of string and case insensitivity is set |
|
func convertEnumCaseStringKind(value interface{}, caseSensitive bool) *string { |
|
if caseSensitive { |
|
return nil |
|
} |
|
|
|
val := reflect.ValueOf(value) |
|
if val.Kind() != reflect.String { |
|
return nil |
|
} |
|
|
|
str := fmt.Sprintf("%v", value) |
|
return &str |
|
} |
|
|
|
// MinItems validates that there are at least n items in a slice |
|
func MinItems(path, in string, size, min int64) *errors.Validation { |
|
if size < min { |
|
return errors.TooFewItems(path, in, min, size) |
|
} |
|
return nil |
|
} |
|
|
|
// MaxItems validates that there are at most n items in a slice |
|
func MaxItems(path, in string, size, max int64) *errors.Validation { |
|
if size > max { |
|
return errors.TooManyItems(path, in, max, size) |
|
} |
|
return nil |
|
} |
|
|
|
// UniqueItems validates that the provided slice has unique elements |
|
func UniqueItems(path, in string, data interface{}) *errors.Validation { |
|
val := reflect.ValueOf(data) |
|
if val.Kind() != reflect.Slice { |
|
return nil |
|
} |
|
var unique []interface{} |
|
for i := 0; i < val.Len(); i++ { |
|
v := val.Index(i).Interface() |
|
for _, u := range unique { |
|
if reflect.DeepEqual(v, u) { |
|
return errors.DuplicateItems(path, in) |
|
} |
|
} |
|
unique = append(unique, v) |
|
} |
|
return nil |
|
} |
|
|
|
// MinLength validates a string for minimum length |
|
func MinLength(path, in, data string, minLength int64) *errors.Validation { |
|
strLen := int64(utf8.RuneCount([]byte(data))) |
|
if strLen < minLength { |
|
return errors.TooShort(path, in, minLength, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MaxLength validates a string for maximum length |
|
func MaxLength(path, in, data string, maxLength int64) *errors.Validation { |
|
strLen := int64(utf8.RuneCount([]byte(data))) |
|
if strLen > maxLength { |
|
return errors.TooLong(path, in, maxLength, data) |
|
} |
|
return nil |
|
} |
|
|
|
// ReadOnly validates an interface for readonly |
|
func ReadOnly(ctx context.Context, path, in string, data interface{}) *errors.Validation { |
|
|
|
// read only is only validated when operationType is request |
|
if op := extractOperationType(ctx); op != request { |
|
return nil |
|
} |
|
|
|
// data must be of zero value of its type |
|
val := reflect.ValueOf(data) |
|
if val.IsValid() { |
|
if reflect.DeepEqual(reflect.Zero(val.Type()).Interface(), val.Interface()) { |
|
return nil |
|
} |
|
} else { |
|
return nil |
|
} |
|
|
|
return errors.ReadOnly(path, in, data) |
|
} |
|
|
|
// Required validates an interface for requiredness |
|
func Required(path, in string, data interface{}) *errors.Validation { |
|
val := reflect.ValueOf(data) |
|
if val.IsValid() { |
|
if reflect.DeepEqual(reflect.Zero(val.Type()).Interface(), val.Interface()) { |
|
return errors.Required(path, in, data) |
|
} |
|
return nil |
|
} |
|
return errors.Required(path, in, data) |
|
} |
|
|
|
// RequiredString validates a string for requiredness |
|
func RequiredString(path, in, data string) *errors.Validation { |
|
if data == "" { |
|
return errors.Required(path, in, data) |
|
} |
|
return nil |
|
} |
|
|
|
// RequiredNumber validates a number for requiredness |
|
func RequiredNumber(path, in string, data float64) *errors.Validation { |
|
if data == 0 { |
|
return errors.Required(path, in, data) |
|
} |
|
return nil |
|
} |
|
|
|
// Pattern validates a string against a regular expression |
|
func Pattern(path, in, data, pattern string) *errors.Validation { |
|
re, err := compileRegexp(pattern) |
|
if err != nil { |
|
return errors.FailedPattern(path, in, fmt.Sprintf("%s, but pattern is invalid: %s", pattern, err.Error()), data) |
|
} |
|
if !re.MatchString(data) { |
|
return errors.FailedPattern(path, in, pattern, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MaximumInt validates if a number is smaller than a given maximum |
|
func MaximumInt(path, in string, data, max int64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data > max) || (exclusive && data >= max) { |
|
return errors.ExceedsMaximumInt(path, in, max, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MaximumUint validates if a number is smaller than a given maximum |
|
func MaximumUint(path, in string, data, max uint64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data > max) || (exclusive && data >= max) { |
|
return errors.ExceedsMaximumUint(path, in, max, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// Maximum validates if a number is smaller than a given maximum |
|
func Maximum(path, in string, data, max float64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data > max) || (exclusive && data >= max) { |
|
return errors.ExceedsMaximum(path, in, max, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// Minimum validates if a number is smaller than a given minimum |
|
func Minimum(path, in string, data, min float64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data < min) || (exclusive && data <= min) { |
|
return errors.ExceedsMinimum(path, in, min, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MinimumInt validates if a number is smaller than a given minimum |
|
func MinimumInt(path, in string, data, min int64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data < min) || (exclusive && data <= min) { |
|
return errors.ExceedsMinimumInt(path, in, min, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MinimumUint validates if a number is smaller than a given minimum |
|
func MinimumUint(path, in string, data, min uint64, exclusive bool) *errors.Validation { |
|
if (!exclusive && data < min) || (exclusive && data <= min) { |
|
return errors.ExceedsMinimumUint(path, in, min, exclusive, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MultipleOf validates if the provided number is a multiple of the factor |
|
func MultipleOf(path, in string, data, factor float64) *errors.Validation { |
|
// multipleOf factor must be positive |
|
if factor < 0 { |
|
return errors.MultipleOfMustBePositive(path, in, factor) |
|
} |
|
var mult float64 |
|
if factor < 1 { |
|
mult = 1 / factor * data |
|
} else { |
|
mult = data / factor |
|
} |
|
if !swag.IsFloat64AJSONInteger(mult) { |
|
return errors.NotMultipleOf(path, in, factor, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MultipleOfInt validates if the provided integer is a multiple of the factor |
|
func MultipleOfInt(path, in string, data int64, factor int64) *errors.Validation { |
|
// multipleOf factor must be positive |
|
if factor < 0 { |
|
return errors.MultipleOfMustBePositive(path, in, factor) |
|
} |
|
mult := data / factor |
|
if mult*factor != data { |
|
return errors.NotMultipleOf(path, in, factor, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MultipleOfUint validates if the provided unsigned integer is a multiple of the factor |
|
func MultipleOfUint(path, in string, data, factor uint64) *errors.Validation { |
|
mult := data / factor |
|
if mult*factor != data { |
|
return errors.NotMultipleOf(path, in, factor, data) |
|
} |
|
return nil |
|
} |
|
|
|
// FormatOf validates if a string matches a format in the format registry |
|
func FormatOf(path, in, format, data string, registry strfmt.Registry) *errors.Validation { |
|
if registry == nil { |
|
registry = strfmt.Default |
|
} |
|
if ok := registry.ContainsName(format); !ok { |
|
return errors.InvalidTypeName(format) |
|
} |
|
if ok := registry.Validates(format, data); !ok { |
|
return errors.InvalidType(path, in, format, data) |
|
} |
|
return nil |
|
} |
|
|
|
// MaximumNativeType provides native type constraint validation as a facade |
|
// to various numeric types versions of Maximum constraint check. |
|
// |
|
// Assumes that any possible loss conversion during conversion has been |
|
// checked beforehand. |
|
// |
|
// NOTE: currently, the max value is marshalled as a float64, no matter what, |
|
// which means there may be a loss during conversions (e.g. for very large integers) |
|
// |
|
// TODO: Normally, a JSON MAX_SAFE_INTEGER check would ensure conversion remains loss-free |
|
func MaximumNativeType(path, in string, val interface{}, max float64, exclusive bool) *errors.Validation { |
|
kind := reflect.ValueOf(val).Type().Kind() |
|
switch kind { |
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: |
|
value := valueHelp.asInt64(val) |
|
return MaximumInt(path, in, value, int64(max), exclusive) |
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |
|
value := valueHelp.asUint64(val) |
|
if max < 0 { |
|
return errors.ExceedsMaximum(path, in, max, exclusive, val) |
|
} |
|
return MaximumUint(path, in, value, uint64(max), exclusive) |
|
case reflect.Float32, reflect.Float64: |
|
fallthrough |
|
default: |
|
value := valueHelp.asFloat64(val) |
|
return Maximum(path, in, value, max, exclusive) |
|
} |
|
} |
|
|
|
// MinimumNativeType provides native type constraint validation as a facade |
|
// to various numeric types versions of Minimum constraint check. |
|
// |
|
// Assumes that any possible loss conversion during conversion has been |
|
// checked beforehand. |
|
// |
|
// NOTE: currently, the min value is marshalled as a float64, no matter what, |
|
// which means there may be a loss during conversions (e.g. for very large integers) |
|
// |
|
// TODO: Normally, a JSON MAX_SAFE_INTEGER check would ensure conversion remains loss-free |
|
func MinimumNativeType(path, in string, val interface{}, min float64, exclusive bool) *errors.Validation { |
|
kind := reflect.ValueOf(val).Type().Kind() |
|
switch kind { |
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: |
|
value := valueHelp.asInt64(val) |
|
return MinimumInt(path, in, value, int64(min), exclusive) |
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |
|
value := valueHelp.asUint64(val) |
|
if min < 0 { |
|
return nil |
|
} |
|
return MinimumUint(path, in, value, uint64(min), exclusive) |
|
case reflect.Float32, reflect.Float64: |
|
fallthrough |
|
default: |
|
value := valueHelp.asFloat64(val) |
|
return Minimum(path, in, value, min, exclusive) |
|
} |
|
} |
|
|
|
// MultipleOfNativeType provides native type constraint validation as a facade |
|
// to various numeric types version of MultipleOf constraint check. |
|
// |
|
// Assumes that any possible loss conversion during conversion has been |
|
// checked beforehand. |
|
// |
|
// NOTE: currently, the multipleOf factor is marshalled as a float64, no matter what, |
|
// which means there may be a loss during conversions (e.g. for very large integers) |
|
// |
|
// TODO: Normally, a JSON MAX_SAFE_INTEGER check would ensure conversion remains loss-free |
|
func MultipleOfNativeType(path, in string, val interface{}, multipleOf float64) *errors.Validation { |
|
kind := reflect.ValueOf(val).Type().Kind() |
|
switch kind { |
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: |
|
value := valueHelp.asInt64(val) |
|
return MultipleOfInt(path, in, value, int64(multipleOf)) |
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |
|
value := valueHelp.asUint64(val) |
|
return MultipleOfUint(path, in, value, uint64(multipleOf)) |
|
case reflect.Float32, reflect.Float64: |
|
fallthrough |
|
default: |
|
value := valueHelp.asFloat64(val) |
|
return MultipleOf(path, in, value, multipleOf) |
|
} |
|
} |
|
|
|
// IsValueValidAgainstRange checks that a numeric value is compatible with |
|
// the range defined by Type and Format, that is, may be converted without loss. |
|
// |
|
// NOTE: this check is about type capacity and not formal verification such as: 1.0 != 1L |
|
func IsValueValidAgainstRange(val interface{}, typeName, format, prefix, path string) error { |
|
kind := reflect.ValueOf(val).Type().Kind() |
|
|
|
// What is the string representation of val |
|
stringRep := "" |
|
switch kind { |
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |
|
stringRep = swag.FormatUint64(valueHelp.asUint64(val)) |
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: |
|
stringRep = swag.FormatInt64(valueHelp.asInt64(val)) |
|
case reflect.Float32, reflect.Float64: |
|
stringRep = swag.FormatFloat64(valueHelp.asFloat64(val)) |
|
default: |
|
return fmt.Errorf("%s value number range checking called with invalid (non numeric) val type in %s", prefix, path) |
|
} |
|
|
|
var errVal error |
|
|
|
switch typeName { |
|
case integerType: |
|
switch format { |
|
case integerFormatInt32: |
|
_, errVal = swag.ConvertInt32(stringRep) |
|
case integerFormatUInt32: |
|
_, errVal = swag.ConvertUint32(stringRep) |
|
case integerFormatUInt64: |
|
_, errVal = swag.ConvertUint64(stringRep) |
|
case integerFormatInt64: |
|
fallthrough |
|
default: |
|
_, errVal = swag.ConvertInt64(stringRep) |
|
} |
|
case numberType: |
|
fallthrough |
|
default: |
|
switch format { |
|
case numberFormatFloat, numberFormatFloat32: |
|
_, errVal = swag.ConvertFloat32(stringRep) |
|
case numberFormatDouble, numberFormatFloat64: |
|
fallthrough |
|
default: |
|
// No check can be performed here since |
|
// no number beyond float64 is supported |
|
} |
|
} |
|
if errVal != nil { // We don't report the actual errVal from strconv |
|
if format != "" { |
|
errVal = fmt.Errorf("%s value must be of type %s with format %s in %s", prefix, typeName, format, path) |
|
} else { |
|
errVal = fmt.Errorf("%s value must be of type %s (default format) in %s", prefix, typeName, path) |
|
} |
|
} |
|
return errVal |
|
}
|
|
|