1
0
Fork 0
mirror of https://github.com/kyverno/kyverno.git synced 2025-03-06 07:57:07 +00:00
kyverno/pkg/engine/pattern.go

309 lines
7.9 KiB
Go
Raw Normal View History

2019-05-27 14:45:54 +03:00
package engine
import (
"math"
2019-05-27 18:07:24 +03:00
"regexp"
"strconv"
2019-05-27 14:45:54 +03:00
"strings"
2019-05-27 18:07:24 +03:00
"github.com/golang/glog"
2019-05-27 18:07:24 +03:00
"github.com/minio/minio/pkg/wildcard"
2019-12-13 13:17:22 -08:00
apiresource "k8s.io/apimachinery/pkg/api/resource"
2019-05-27 18:07:24 +03:00
)
// Operator is string alias that represents selection operators enum
type Operator string
const (
// Equal stands for ==
Equal Operator = ""
// MoreEqual stands for >=
MoreEqual Operator = ">="
// LessEqual stands for <=
LessEqual Operator = "<="
// NotEqual stands for !
NotEqual Operator = "!"
// More stands for >
More Operator = ">"
// Less stands for <
Less Operator = "<"
2019-05-27 14:45:54 +03:00
)
const relativePrefix Operator = "./"
const referenceSign Operator = "$()"
2019-12-13 13:17:22 -08:00
type quantity int
const (
equal quantity = 0
lessThan quantity = -1
greaterThan quantity = 1
)
2019-05-27 14:45:54 +03:00
// ValidateValueWithPattern validates value with operators and wildcards
func ValidateValueWithPattern(value, pattern interface{}) bool {
switch typedPattern := pattern.(type) {
case bool:
typedValue, ok := value.(bool)
if !ok {
glog.V(4).Infof("Expected bool, found %T", value)
2019-05-27 14:45:54 +03:00
return false
}
return typedPattern == typedValue
case int:
2019-05-27 18:07:24 +03:00
return validateValueWithIntPattern(value, int64(typedPattern))
case int64:
2019-05-27 14:45:54 +03:00
return validateValueWithIntPattern(value, typedPattern)
case float64:
return validateValueWithFloatPattern(value, typedPattern)
case string:
2019-05-27 18:07:24 +03:00
return validateValueWithStringPatterns(value, typedPattern)
case nil:
return validateValueWithNilPattern(value)
2019-08-27 11:41:47 -07:00
case map[string]interface{}:
2019-09-25 16:06:37 -07:00
// TODO: check if this is ever called?
2019-08-27 11:41:47 -07:00
return validateValueWithMapPattern(value, typedPattern)
case []interface{}:
2019-09-25 16:06:37 -07:00
// TODO: check if this is ever called?
2019-08-27 11:41:47 -07:00
glog.Warning("Arrays as patterns are not supported")
2019-05-27 14:45:54 +03:00
return false
default:
glog.Warningf("Unknown type as pattern: %v", typedPattern)
2019-05-27 14:45:54 +03:00
return false
}
}
2019-08-27 11:41:47 -07:00
func validateValueWithMapPattern(value interface{}, typedPattern map[string]interface{}) bool {
// verify the type of the resource value is map[string]interface,
// we only check for existance of object, not the equality of content and value
2019-09-25 16:06:37 -07:00
//TODO: check if adding
2019-08-27 11:41:47 -07:00
_, ok := value.(map[string]interface{})
if !ok {
glog.Warningf("Expected map[string]interface{}, found %T\n", value)
return false
}
return true
}
2019-06-10 17:32:26 +03:00
// Handler for int values during validation process
2019-05-27 18:07:24 +03:00
func validateValueWithIntPattern(value interface{}, pattern int64) bool {
2019-05-27 14:45:54 +03:00
switch typedValue := value.(type) {
case int:
2019-05-27 18:07:24 +03:00
return int64(typedValue) == pattern
case int64:
2019-05-27 14:45:54 +03:00
return typedValue == pattern
case float64:
// check that float has no fraction
if typedValue == math.Trunc(typedValue) {
2019-05-27 18:07:24 +03:00
return int64(typedValue) == pattern
2019-05-27 14:45:54 +03:00
}
glog.Warningf("Expected int, found float: %f\n", typedValue)
2019-05-27 14:45:54 +03:00
return false
case string:
// extract int64 from string
int64Num, err := strconv.ParseInt(typedValue, 10, 64)
if err != nil {
glog.Warningf("Failed to parse int64 from string: %v", err)
return false
}
return int64Num == pattern
2019-05-27 14:45:54 +03:00
default:
glog.Warningf("Expected int, found: %T\n", value)
2019-05-27 14:45:54 +03:00
return false
}
}
2019-06-10 17:32:26 +03:00
// Handler for float values during validation process
2019-05-27 14:45:54 +03:00
func validateValueWithFloatPattern(value interface{}, pattern float64) bool {
switch typedValue := value.(type) {
case int:
// check that float has no fraction
if pattern == math.Trunc(pattern) {
return int(pattern) == value
}
glog.Warningf("Expected float, found int: %d\n", typedValue)
return false
case int64:
// check that float has no fraction
if pattern == math.Trunc(pattern) {
return int64(pattern) == value
}
glog.Warningf("Expected float, found int: %d\n", typedValue)
2019-05-27 14:45:54 +03:00
return false
case float64:
return typedValue == pattern
case string:
// extract float64 from string
float64Num, err := strconv.ParseFloat(typedValue, 64)
if err != nil {
glog.Warningf("Failed to parse float64 from string: %v", err)
return false
}
return float64Num == pattern
2019-05-27 14:45:54 +03:00
default:
glog.Warningf("Expected float, found: %T\n", value)
2019-05-27 14:45:54 +03:00
return false
}
}
2019-06-10 17:32:26 +03:00
// Handler for nil values during validation process
2019-05-27 14:45:54 +03:00
func validateValueWithNilPattern(value interface{}) bool {
switch typed := value.(type) {
case float64:
return typed == 0.0
case int:
return typed == 0
2019-05-27 18:07:24 +03:00
case int64:
return typed == 0
2019-05-27 14:45:54 +03:00
case string:
return typed == ""
case bool:
return typed == false
case nil:
return true
case map[string]interface{}, []interface{}:
glog.Warningf("Maps and arrays could not be checked with nil pattern")
2019-05-27 14:45:54 +03:00
return false
default:
glog.Warningf("Unknown type as value when checking for nil pattern: %T\n", value)
2019-05-27 14:45:54 +03:00
return false
}
}
2019-06-10 17:32:26 +03:00
// Handler for pattern values during validation process
2019-05-27 18:07:24 +03:00
func validateValueWithStringPatterns(value interface{}, pattern string) bool {
2019-05-27 14:45:54 +03:00
statements := strings.Split(pattern, "|")
2019-05-27 18:07:24 +03:00
for _, statement := range statements {
statement = strings.Trim(statement, " ")
if validateValueWithStringPattern(value, statement) {
2019-05-27 14:45:54 +03:00
return true
}
}
return false
}
2019-06-10 17:32:26 +03:00
// Handler for single pattern value during validation process
// Detects if pattern has a number
2019-05-27 18:07:24 +03:00
func validateValueWithStringPattern(value interface{}, pattern string) bool {
operator := getOperatorFromStringPattern(pattern)
pattern = pattern[len(operator):]
number, str := getNumberAndStringPartsFromPattern(pattern)
if "" == number {
return validateString(value, str, operator)
}
2019-12-13 13:17:22 -08:00
return validateNumberWithStr(value, pattern, operator)
2019-05-27 14:45:54 +03:00
}
2019-05-27 18:07:24 +03:00
2019-06-10 17:32:26 +03:00
// Handler for string values
2019-05-27 18:07:24 +03:00
func validateString(value interface{}, pattern string, operator Operator) bool {
if NotEqual == operator || Equal == operator {
strValue, ok := value.(string)
if !ok {
glog.Warningf("Expected string, found %T\n", value)
2019-05-27 18:07:24 +03:00
return false
}
wildcardResult := wildcard.Match(pattern, strValue)
if NotEqual == operator {
return !wildcardResult
}
return wildcardResult
}
glog.Warningf("Operators >, >=, <, <= are not applicable to strings")
2019-05-27 18:07:24 +03:00
return false
}
2019-12-13 13:17:22 -08:00
// validateNumberWithStr compares quantity if pattern type is quantity
// or a wildcard match to pattern string
func validateNumberWithStr(value interface{}, pattern string, operator Operator) bool {
typedValue, err := convertToString(value)
if err != nil {
glog.Warning(err)
return false
}
2019-05-27 18:07:24 +03:00
2019-12-13 13:17:22 -08:00
patternQuan, err := apiresource.ParseQuantity(pattern)
// 1. nil error - quantity comparison
if err == nil {
valueQuan, err := apiresource.ParseQuantity(typedValue)
if err != nil {
glog.Warningf("Invalid quantity in resource %s, err: %v\n", typedValue, err)
2019-05-27 18:07:24 +03:00
return false
}
2019-12-13 13:17:22 -08:00
return compareQuantity(valueQuan, patternQuan, operator)
2019-05-27 18:07:24 +03:00
}
2019-12-13 13:17:22 -08:00
// 2. wildcard match
if !wildcard.Match(pattern, typedValue) {
glog.Warningf("Value '%s' has not passed wildcard check: %s", typedValue, pattern)
2019-05-27 18:07:24 +03:00
return false
}
2019-12-13 13:17:22 -08:00
return true
}
2019-05-27 18:07:24 +03:00
2019-12-13 13:17:22 -08:00
func compareQuantity(value, pattern apiresource.Quantity, operator Operator) bool {
result := value.Cmp(pattern)
2019-05-27 18:07:24 +03:00
switch operator {
case Equal:
2019-12-13 13:17:22 -08:00
return result == int(equal)
2019-05-27 18:07:24 +03:00
case NotEqual:
2019-12-13 13:17:22 -08:00
return result != int(equal)
2019-05-27 18:07:24 +03:00
case More:
2019-12-13 13:17:22 -08:00
return result == int(greaterThan)
2019-05-27 18:07:24 +03:00
case Less:
2019-12-13 13:17:22 -08:00
return result == int(lessThan)
case MoreEqual:
return (result == int(equal)) || (result == int(greaterThan))
2019-05-27 18:07:24 +03:00
case LessEqual:
2019-12-13 13:17:22 -08:00
return (result == int(equal)) || (result == int(lessThan))
2019-05-27 18:07:24 +03:00
}
return false
}
2019-06-10 17:32:26 +03:00
// getOperatorFromStringPattern parses opeartor from pattern
2019-05-27 18:07:24 +03:00
func getOperatorFromStringPattern(pattern string) Operator {
if len(pattern) < 2 {
return Equal
}
2019-05-27 18:07:24 +03:00
if pattern[:len(MoreEqual)] == string(MoreEqual) {
return MoreEqual
}
if pattern[:len(LessEqual)] == string(LessEqual) {
return LessEqual
}
if pattern[:len(More)] == string(More) {
return More
}
if pattern[:len(Less)] == string(Less) {
return Less
}
if pattern[:len(NotEqual)] == string(NotEqual) {
return NotEqual
}
return Equal
2019-05-27 18:07:24 +03:00
}
2019-06-10 17:32:26 +03:00
// detects numerical and string parts in pattern and returns them
2019-05-27 18:07:24 +03:00
func getNumberAndStringPartsFromPattern(pattern string) (number, str string) {
regexpStr := `^(\d*(\.\d+)?)(.*)`
re := regexp.MustCompile(regexpStr)
matches := re.FindAllStringSubmatch(pattern, -1)
match := matches[0]
return match[1], match[3]
}