mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-25 18:09:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			199 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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 (
 | |
| 	"log"
 | |
| 	"reflect"
 | |
| 	"regexp"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/go-openapi/errors"
 | |
| 	"github.com/go-openapi/spec"
 | |
| 	"github.com/go-openapi/strfmt"
 | |
| )
 | |
| 
 | |
| type objectValidator struct {
 | |
| 	Path                 string
 | |
| 	In                   string
 | |
| 	MaxProperties        *int64
 | |
| 	MinProperties        *int64
 | |
| 	Required             []string
 | |
| 	Properties           map[string]spec.Schema
 | |
| 	AdditionalProperties *spec.SchemaOrBool
 | |
| 	PatternProperties    map[string]spec.Schema
 | |
| 	Root                 interface{}
 | |
| 	KnownFormats         strfmt.Registry
 | |
| }
 | |
| 
 | |
| func (o *objectValidator) SetPath(path string) {
 | |
| 	o.Path = path
 | |
| }
 | |
| 
 | |
| func (o *objectValidator) Applies(source interface{}, kind reflect.Kind) bool {
 | |
| 	// TODO: this should also work for structs
 | |
| 	// there is a problem in the type validator where it will be unhappy about null values
 | |
| 	// so that requires more testing
 | |
| 	r := reflect.TypeOf(source) == specSchemaType && (kind == reflect.Map || kind == reflect.Struct)
 | |
| 	if Debug {
 | |
| 		log.Printf("object validator for %q applies %t for %T (kind: %v)\n", o.Path, r, source, kind)
 | |
| 	}
 | |
| 	return r
 | |
| }
 | |
| 
 | |
| func (o *objectValidator) isPropertyName() bool {
 | |
| 	p := strings.Split(o.Path, ".")
 | |
| 	return p[len(p)-1] == "properties" && p[len(p)-2] != "properties"
 | |
| }
 | |
| func (o *objectValidator) checkArrayMustHaveItems(res *Result, val map[string]interface{}) {
 | |
| 	if t, typeFound := val["type"]; typeFound {
 | |
| 		if tpe, ok := t.(string); ok && tpe == "array" {
 | |
| 			if _, itemsKeyFound := val["items"]; !itemsKeyFound {
 | |
| 				res.AddErrors(errors.Required("items", o.Path))
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (o *objectValidator) checkItemsMustBeTypeArray(res *Result, val map[string]interface{}) {
 | |
| 	if !o.isPropertyName() {
 | |
| 		if _, itemsKeyFound := val["items"]; itemsKeyFound {
 | |
| 			t, typeFound := val["type"]
 | |
| 			if typeFound {
 | |
| 				if tpe, ok := t.(string); !ok || tpe != "array" {
 | |
| 					res.AddErrors(errors.InvalidType(o.Path, o.In, "array", nil))
 | |
| 				}
 | |
| 			} else {
 | |
| 				// there is no type
 | |
| 				res.AddErrors(errors.Required("type", o.Path))
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| func (o *objectValidator) precheck(res *Result, val map[string]interface{}) {
 | |
| 	o.checkArrayMustHaveItems(res, val)
 | |
| 	o.checkItemsMustBeTypeArray(res, val)
 | |
| }
 | |
| func (o *objectValidator) Validate(data interface{}) *Result {
 | |
| 	val := data.(map[string]interface{})
 | |
| 	numKeys := int64(len(val))
 | |
| 
 | |
| 	if o.MinProperties != nil && numKeys < *o.MinProperties {
 | |
| 		return sErr(errors.TooFewProperties(o.Path, o.In, *o.MinProperties))
 | |
| 	}
 | |
| 	if o.MaxProperties != nil && numKeys > *o.MaxProperties {
 | |
| 		return sErr(errors.TooManyProperties(o.Path, o.In, *o.MaxProperties))
 | |
| 	}
 | |
| 
 | |
| 	res := new(Result)
 | |
| 
 | |
| 	o.precheck(res, val)
 | |
| 
 | |
| 	if o.AdditionalProperties != nil && !o.AdditionalProperties.Allows {
 | |
| 		for k := range val {
 | |
| 			_, regularProperty := o.Properties[k]
 | |
| 			matched := false
 | |
| 
 | |
| 			for pk := range o.PatternProperties {
 | |
| 				if matches, _ := regexp.MatchString(pk, k); matches {
 | |
| 					matched = true
 | |
| 					break
 | |
| 				}
 | |
| 			}
 | |
| 			if !regularProperty && k != "$schema" && k != "id" && !matched {
 | |
| 				res.AddErrors(errors.PropertyNotAllowed(o.Path, o.In, k))
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		for key, value := range val {
 | |
| 			_, regularProperty := o.Properties[key]
 | |
| 			matched, succeededOnce, _ := o.validatePatternProperty(key, value, res)
 | |
| 			if !(regularProperty || matched || succeededOnce) {
 | |
| 				if o.AdditionalProperties != nil && o.AdditionalProperties.Schema != nil {
 | |
| 					res.Merge(NewSchemaValidator(o.AdditionalProperties.Schema, o.Root, o.Path+"."+key, o.KnownFormats).Validate(value))
 | |
| 				} else if regularProperty && !(matched || succeededOnce) {
 | |
| 					res.AddErrors(errors.FailedAllPatternProperties(o.Path, o.In, key))
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	createdFromDefaults := map[string]bool{}
 | |
| 
 | |
| 	for pName, pSchema := range o.Properties {
 | |
| 		rName := pName
 | |
| 		if o.Path != "" {
 | |
| 			rName = o.Path + "." + pName
 | |
| 		}
 | |
| 
 | |
| 		if v, ok := val[pName]; ok {
 | |
| 			r := NewSchemaValidator(&pSchema, o.Root, rName, o.KnownFormats).Validate(v)
 | |
| 			res.Merge(r)
 | |
| 		} else if pSchema.Default != nil {
 | |
| 			createdFromDefaults[pName] = true
 | |
| 			pName := pName // shaddow
 | |
| 			def := pSchema.Default
 | |
| 			res.Defaulters = append(res.Defaulters, DefaulterFunc(func() {
 | |
| 				val[pName] = def
 | |
| 			}))
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if len(o.Required) > 0 {
 | |
| 		for _, k := range o.Required {
 | |
| 			if _, ok := val[k]; !ok && !createdFromDefaults[k] {
 | |
| 				res.AddErrors(errors.Required(o.Path+"."+k, o.In))
 | |
| 				continue
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for key, value := range val {
 | |
| 		_, regularProperty := o.Properties[key]
 | |
| 		matched, succeededOnce, patterns := o.validatePatternProperty(key, value, res)
 | |
| 		if !regularProperty && (matched || succeededOnce) {
 | |
| 			for _, pName := range patterns {
 | |
| 				if v, ok := o.PatternProperties[pName]; ok {
 | |
| 					res.Merge(NewSchemaValidator(&v, o.Root, o.Path+"."+key, o.KnownFormats).Validate(value))
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func (o *objectValidator) validatePatternProperty(key string, value interface{}, result *Result) (bool, bool, []string) {
 | |
| 	matched := false
 | |
| 	succeededOnce := false
 | |
| 	var patterns []string
 | |
| 
 | |
| 	for k, schema := range o.PatternProperties {
 | |
| 		if match, _ := regexp.MatchString(k, key); match {
 | |
| 			patterns = append(patterns, k)
 | |
| 			matched = true
 | |
| 			validator := NewSchemaValidator(&schema, o.Root, o.Path+"."+key, o.KnownFormats)
 | |
| 
 | |
| 			res := validator.Validate(value)
 | |
| 			result.Merge(res)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if succeededOnce {
 | |
| 		result.Inc()
 | |
| 	}
 | |
| 
 | |
| 	return matched, succeededOnce, patterns
 | |
| }
 |