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.
598 lines
16 KiB
598 lines
16 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 errors
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
invalidType = "%s is an invalid type name"
|
|
typeFail = "%s in %s must be of type %s"
|
|
typeFailWithData = "%s in %s must be of type %s: %q"
|
|
typeFailWithError = "%s in %s must be of type %s, because: %s"
|
|
requiredFail = "%s in %s is required"
|
|
readOnlyFail = "%s in %s is readOnly"
|
|
tooLongMessage = "%s in %s should be at most %d chars long"
|
|
tooShortMessage = "%s in %s should be at least %d chars long"
|
|
patternFail = "%s in %s should match '%s'"
|
|
enumFail = "%s in %s should be one of %v"
|
|
multipleOfFail = "%s in %s should be a multiple of %v"
|
|
maxIncFail = "%s in %s should be less than or equal to %v"
|
|
maxExcFail = "%s in %s should be less than %v"
|
|
minIncFail = "%s in %s should be greater than or equal to %v"
|
|
minExcFail = "%s in %s should be greater than %v"
|
|
uniqueFail = "%s in %s shouldn't contain duplicates"
|
|
maxItemsFail = "%s in %s should have at most %d items"
|
|
minItemsFail = "%s in %s should have at least %d items"
|
|
typeFailNoIn = "%s must be of type %s"
|
|
typeFailWithDataNoIn = "%s must be of type %s: %q"
|
|
typeFailWithErrorNoIn = "%s must be of type %s, because: %s"
|
|
requiredFailNoIn = "%s is required"
|
|
readOnlyFailNoIn = "%s is readOnly"
|
|
tooLongMessageNoIn = "%s should be at most %d chars long"
|
|
tooShortMessageNoIn = "%s should be at least %d chars long"
|
|
patternFailNoIn = "%s should match '%s'"
|
|
enumFailNoIn = "%s should be one of %v"
|
|
multipleOfFailNoIn = "%s should be a multiple of %v"
|
|
maxIncFailNoIn = "%s should be less than or equal to %v"
|
|
maxExcFailNoIn = "%s should be less than %v"
|
|
minIncFailNoIn = "%s should be greater than or equal to %v"
|
|
minExcFailNoIn = "%s should be greater than %v"
|
|
uniqueFailNoIn = "%s shouldn't contain duplicates"
|
|
maxItemsFailNoIn = "%s should have at most %d items"
|
|
minItemsFailNoIn = "%s should have at least %d items"
|
|
noAdditionalItems = "%s in %s can't have additional items"
|
|
noAdditionalItemsNoIn = "%s can't have additional items"
|
|
tooFewProperties = "%s in %s should have at least %d properties"
|
|
tooFewPropertiesNoIn = "%s should have at least %d properties"
|
|
tooManyProperties = "%s in %s should have at most %d properties"
|
|
tooManyPropertiesNoIn = "%s should have at most %d properties"
|
|
unallowedProperty = "%s.%s in %s is a forbidden property"
|
|
unallowedPropertyNoIn = "%s.%s is a forbidden property"
|
|
failedAllPatternProps = "%s.%s in %s failed all pattern properties"
|
|
failedAllPatternPropsNoIn = "%s.%s failed all pattern properties"
|
|
multipleOfMustBePositive = "factor MultipleOf declared for %s must be positive: %v"
|
|
)
|
|
|
|
// All code responses can be used to differentiate errors for different handling
|
|
// by the consuming program
|
|
const (
|
|
// CompositeErrorCode remains 422 for backwards-compatibility
|
|
// and to separate it from validation errors with cause
|
|
CompositeErrorCode = 422
|
|
// InvalidTypeCode is used for any subclass of invalid types
|
|
InvalidTypeCode = 600 + iota
|
|
RequiredFailCode
|
|
TooLongFailCode
|
|
TooShortFailCode
|
|
PatternFailCode
|
|
EnumFailCode
|
|
MultipleOfFailCode
|
|
MaxFailCode
|
|
MinFailCode
|
|
UniqueFailCode
|
|
MaxItemsFailCode
|
|
MinItemsFailCode
|
|
NoAdditionalItemsCode
|
|
TooFewPropertiesCode
|
|
TooManyPropertiesCode
|
|
UnallowedPropertyCode
|
|
FailedAllPatternPropsCode
|
|
MultipleOfMustBePositiveCode
|
|
ReadOnlyFailCode
|
|
)
|
|
|
|
// CompositeError is an error that groups several errors together
|
|
type CompositeError struct {
|
|
Errors []error
|
|
code int32
|
|
message string
|
|
}
|
|
|
|
// Code for this error
|
|
func (c *CompositeError) Code() int32 {
|
|
return c.code
|
|
}
|
|
|
|
func (c *CompositeError) Error() string {
|
|
if len(c.Errors) > 0 {
|
|
msgs := []string{c.message + ":"}
|
|
for _, e := range c.Errors {
|
|
msgs = append(msgs, e.Error())
|
|
}
|
|
return strings.Join(msgs, "\n")
|
|
}
|
|
return c.message
|
|
}
|
|
|
|
// MarshalJSON implements the JSON encoding interface
|
|
func (c CompositeError) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(map[string]interface{}{
|
|
"code": c.code,
|
|
"message": c.message,
|
|
"errors": c.Errors,
|
|
})
|
|
}
|
|
|
|
// CompositeValidationError an error to wrap a bunch of other errors
|
|
func CompositeValidationError(errors ...error) *CompositeError {
|
|
return &CompositeError{
|
|
code: CompositeErrorCode,
|
|
Errors: append([]error{}, errors...),
|
|
message: "validation failure list",
|
|
}
|
|
}
|
|
|
|
// FailedAllPatternProperties an error for when the property doesn't match a pattern
|
|
func FailedAllPatternProperties(name, in, key string) *Validation {
|
|
msg := fmt.Sprintf(failedAllPatternProps, name, key, in)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(failedAllPatternPropsNoIn, name, key)
|
|
}
|
|
return &Validation{
|
|
code: FailedAllPatternPropsCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: key,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// PropertyNotAllowed an error for when the property doesn't match a pattern
|
|
func PropertyNotAllowed(name, in, key string) *Validation {
|
|
msg := fmt.Sprintf(unallowedProperty, name, key, in)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(unallowedPropertyNoIn, name, key)
|
|
}
|
|
return &Validation{
|
|
code: UnallowedPropertyCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: key,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooFewProperties an error for an object with too few properties
|
|
func TooFewProperties(name, in string, n int64) *Validation {
|
|
msg := fmt.Sprintf(tooFewProperties, name, in, n)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(tooFewPropertiesNoIn, name, n)
|
|
}
|
|
return &Validation{
|
|
code: TooFewPropertiesCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: n,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooManyProperties an error for an object with too many properties
|
|
func TooManyProperties(name, in string, n int64) *Validation {
|
|
msg := fmt.Sprintf(tooManyProperties, name, in, n)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(tooManyPropertiesNoIn, name, n)
|
|
}
|
|
return &Validation{
|
|
code: TooManyPropertiesCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: n,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// AdditionalItemsNotAllowed an error for invalid additional items
|
|
func AdditionalItemsNotAllowed(name, in string) *Validation {
|
|
msg := fmt.Sprintf(noAdditionalItems, name, in)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(noAdditionalItemsNoIn, name)
|
|
}
|
|
return &Validation{
|
|
code: NoAdditionalItemsCode,
|
|
Name: name,
|
|
In: in,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// InvalidCollectionFormat another flavor of invalid type error
|
|
func InvalidCollectionFormat(name, in, format string) *Validation {
|
|
return &Validation{
|
|
code: InvalidTypeCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: format,
|
|
message: fmt.Sprintf("the collection format %q is not supported for the %s param %q", format, in, name),
|
|
}
|
|
}
|
|
|
|
// InvalidTypeName an error for when the type is invalid
|
|
func InvalidTypeName(typeName string) *Validation {
|
|
return &Validation{
|
|
code: InvalidTypeCode,
|
|
Value: typeName,
|
|
message: fmt.Sprintf(invalidType, typeName),
|
|
}
|
|
}
|
|
|
|
// InvalidType creates an error for when the type is invalid
|
|
func InvalidType(name, in, typeName string, value interface{}) *Validation {
|
|
var message string
|
|
|
|
if in != "" {
|
|
switch value.(type) {
|
|
case string:
|
|
message = fmt.Sprintf(typeFailWithData, name, in, typeName, value)
|
|
case error:
|
|
message = fmt.Sprintf(typeFailWithError, name, in, typeName, value)
|
|
default:
|
|
message = fmt.Sprintf(typeFail, name, in, typeName)
|
|
}
|
|
} else {
|
|
switch value.(type) {
|
|
case string:
|
|
message = fmt.Sprintf(typeFailWithDataNoIn, name, typeName, value)
|
|
case error:
|
|
message = fmt.Sprintf(typeFailWithErrorNoIn, name, typeName, value)
|
|
default:
|
|
message = fmt.Sprintf(typeFailNoIn, name, typeName)
|
|
}
|
|
}
|
|
|
|
return &Validation{
|
|
code: InvalidTypeCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
|
|
}
|
|
|
|
// DuplicateItems error for when an array contains duplicates
|
|
func DuplicateItems(name, in string) *Validation {
|
|
msg := fmt.Sprintf(uniqueFail, name, in)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(uniqueFailNoIn, name)
|
|
}
|
|
return &Validation{
|
|
code: UniqueFailCode,
|
|
Name: name,
|
|
In: in,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooManyItems error for when an array contains too many items
|
|
func TooManyItems(name, in string, max int64, value interface{}) *Validation {
|
|
msg := fmt.Sprintf(maxItemsFail, name, in, max)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(maxItemsFailNoIn, name, max)
|
|
}
|
|
|
|
return &Validation{
|
|
code: MaxItemsFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooFewItems error for when an array contains too few items
|
|
func TooFewItems(name, in string, min int64, value interface{}) *Validation {
|
|
msg := fmt.Sprintf(minItemsFail, name, in, min)
|
|
if in == "" {
|
|
msg = fmt.Sprintf(minItemsFailNoIn, name, min)
|
|
}
|
|
return &Validation{
|
|
code: MinItemsFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// ExceedsMaximumInt error for when maximum validation fails
|
|
func ExceedsMaximumInt(name, in string, max int64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := maxIncFailNoIn
|
|
if exclusive {
|
|
m = maxExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, max)
|
|
} else {
|
|
m := maxIncFail
|
|
if exclusive {
|
|
m = maxExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, max)
|
|
}
|
|
return &Validation{
|
|
code: MaxFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// ExceedsMaximumUint error for when maximum validation fails
|
|
func ExceedsMaximumUint(name, in string, max uint64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := maxIncFailNoIn
|
|
if exclusive {
|
|
m = maxExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, max)
|
|
} else {
|
|
m := maxIncFail
|
|
if exclusive {
|
|
m = maxExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, max)
|
|
}
|
|
return &Validation{
|
|
code: MaxFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// ExceedsMaximum error for when maximum validation fails
|
|
func ExceedsMaximum(name, in string, max float64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := maxIncFailNoIn
|
|
if exclusive {
|
|
m = maxExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, max)
|
|
} else {
|
|
m := maxIncFail
|
|
if exclusive {
|
|
m = maxExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, max)
|
|
}
|
|
return &Validation{
|
|
code: MaxFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// ExceedsMinimumInt error for when minimum validation fails
|
|
func ExceedsMinimumInt(name, in string, min int64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := minIncFailNoIn
|
|
if exclusive {
|
|
m = minExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, min)
|
|
} else {
|
|
m := minIncFail
|
|
if exclusive {
|
|
m = minExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, min)
|
|
}
|
|
return &Validation{
|
|
code: MinFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// ExceedsMinimumUint error for when minimum validation fails
|
|
func ExceedsMinimumUint(name, in string, min uint64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := minIncFailNoIn
|
|
if exclusive {
|
|
m = minExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, min)
|
|
} else {
|
|
m := minIncFail
|
|
if exclusive {
|
|
m = minExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, min)
|
|
}
|
|
return &Validation{
|
|
code: MinFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// ExceedsMinimum error for when minimum validation fails
|
|
func ExceedsMinimum(name, in string, min float64, exclusive bool, value interface{}) *Validation {
|
|
var message string
|
|
if in == "" {
|
|
m := minIncFailNoIn
|
|
if exclusive {
|
|
m = minExcFailNoIn
|
|
}
|
|
message = fmt.Sprintf(m, name, min)
|
|
} else {
|
|
m := minIncFail
|
|
if exclusive {
|
|
m = minExcFail
|
|
}
|
|
message = fmt.Sprintf(m, name, in, min)
|
|
}
|
|
return &Validation{
|
|
code: MinFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
// NotMultipleOf error for when multiple of validation fails
|
|
func NotMultipleOf(name, in string, multiple, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(multipleOfFailNoIn, name, multiple)
|
|
} else {
|
|
msg = fmt.Sprintf(multipleOfFail, name, in, multiple)
|
|
}
|
|
return &Validation{
|
|
code: MultipleOfFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// EnumFail error for when an enum validation fails
|
|
func EnumFail(name, in string, value interface{}, values []interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(enumFailNoIn, name, values)
|
|
} else {
|
|
msg = fmt.Sprintf(enumFail, name, in, values)
|
|
}
|
|
|
|
return &Validation{
|
|
code: EnumFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
Values: values,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// Required error for when a value is missing
|
|
func Required(name, in string, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(requiredFailNoIn, name)
|
|
} else {
|
|
msg = fmt.Sprintf(requiredFail, name, in)
|
|
}
|
|
return &Validation{
|
|
code: RequiredFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// ReadOnly error for when a value is present in request
|
|
func ReadOnly(name, in string, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(readOnlyFailNoIn, name)
|
|
} else {
|
|
msg = fmt.Sprintf(readOnlyFail, name, in)
|
|
}
|
|
return &Validation{
|
|
code: ReadOnlyFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooLong error for when a string is too long
|
|
func TooLong(name, in string, max int64, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(tooLongMessageNoIn, name, max)
|
|
} else {
|
|
msg = fmt.Sprintf(tooLongMessage, name, in, max)
|
|
}
|
|
return &Validation{
|
|
code: TooLongFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// TooShort error for when a string is too short
|
|
func TooShort(name, in string, min int64, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(tooShortMessageNoIn, name, min)
|
|
} else {
|
|
msg = fmt.Sprintf(tooShortMessage, name, in, min)
|
|
}
|
|
|
|
return &Validation{
|
|
code: TooShortFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// FailedPattern error for when a string fails a regex pattern match
|
|
// the pattern that is returned is the ECMA syntax version of the pattern not the golang version.
|
|
func FailedPattern(name, in, pattern string, value interface{}) *Validation {
|
|
var msg string
|
|
if in == "" {
|
|
msg = fmt.Sprintf(patternFailNoIn, name, pattern)
|
|
} else {
|
|
msg = fmt.Sprintf(patternFail, name, in, pattern)
|
|
}
|
|
|
|
return &Validation{
|
|
code: PatternFailCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: value,
|
|
message: msg,
|
|
}
|
|
}
|
|
|
|
// MultipleOfMustBePositive error for when a
|
|
// multipleOf factor is negative
|
|
func MultipleOfMustBePositive(name, in string, factor interface{}) *Validation {
|
|
return &Validation{
|
|
code: MultipleOfMustBePositiveCode,
|
|
Name: name,
|
|
In: in,
|
|
Value: factor,
|
|
message: fmt.Sprintf(multipleOfMustBePositive, name, factor),
|
|
}
|
|
}
|
|
|