2018-04-25 04:23:30 +05:30
|
|
|
/*
|
2019-04-09 11:39:42 -07:00
|
|
|
* MinIO Cloud Storage, (C) 2018 MinIO, Inc.
|
2018-04-25 04:23:30 +05:30
|
|
|
*
|
|
|
|
* 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 condition
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Function - condition function interface.
|
|
|
|
type Function interface {
|
|
|
|
// evaluate() - evaluates this condition function with given values.
|
|
|
|
evaluate(values map[string][]string) bool
|
|
|
|
|
|
|
|
// key() - returns condition key used in this function.
|
|
|
|
key() Key
|
|
|
|
|
|
|
|
// name() - returns condition name of this function.
|
|
|
|
name() name
|
|
|
|
|
|
|
|
// String() - returns string representation of function.
|
|
|
|
String() string
|
|
|
|
|
|
|
|
// toMap - returns map representation of this function.
|
|
|
|
toMap() map[Key]ValueSet
|
|
|
|
}
|
|
|
|
|
|
|
|
// Functions - list of functions.
|
|
|
|
type Functions []Function
|
|
|
|
|
|
|
|
// Evaluate - evaluates all functions with given values map. Each function is evaluated
|
|
|
|
// sequencely and next function is called only if current function succeeds.
|
|
|
|
func (functions Functions) Evaluate(values map[string][]string) bool {
|
|
|
|
for _, f := range functions {
|
|
|
|
if !f.evaluate(values) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keys - returns list of keys used in all functions.
|
|
|
|
func (functions Functions) Keys() KeySet {
|
|
|
|
keySet := NewKeySet()
|
|
|
|
|
|
|
|
for _, f := range functions {
|
|
|
|
keySet.Add(f.key())
|
|
|
|
}
|
|
|
|
|
|
|
|
return keySet
|
|
|
|
}
|
|
|
|
|
2021-03-16 16:50:36 +01:00
|
|
|
// Clone clones Functions structure
|
|
|
|
func (functions Functions) Clone() Functions {
|
|
|
|
funcs := []Function{}
|
|
|
|
|
|
|
|
for _, f := range functions {
|
|
|
|
vfn := conditionFuncMap[f.name()]
|
|
|
|
for key, values := range f.toMap() {
|
|
|
|
function, _ := vfn(key, values.Clone())
|
|
|
|
funcs = append(funcs, function)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return funcs
|
|
|
|
}
|
|
|
|
|
|
|
|
// Equals returns true if two Functions structures are equal
|
|
|
|
func (functions Functions) Equals(funcs Functions) bool {
|
|
|
|
if len(functions) != len(funcs) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for _, fi := range functions {
|
|
|
|
fistr := fi.String()
|
|
|
|
found := false
|
|
|
|
for _, fj := range funcs {
|
|
|
|
if fistr == fj.String() {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2018-04-25 04:23:30 +05:30
|
|
|
// MarshalJSON - encodes Functions to JSON data.
|
|
|
|
func (functions Functions) MarshalJSON() ([]byte, error) {
|
|
|
|
nm := make(map[name]map[Key]ValueSet)
|
|
|
|
|
|
|
|
for _, f := range functions {
|
2019-01-17 23:58:24 +05:30
|
|
|
if _, ok := nm[f.name()]; ok {
|
|
|
|
for k, v := range f.toMap() {
|
|
|
|
nm[f.name()][k] = v
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
nm[f.name()] = f.toMap()
|
|
|
|
}
|
2018-04-25 04:23:30 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(nm)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (functions Functions) String() string {
|
|
|
|
funcStrings := []string{}
|
|
|
|
for _, f := range functions {
|
|
|
|
s := fmt.Sprintf("%v", f)
|
|
|
|
funcStrings = append(funcStrings, s)
|
|
|
|
}
|
|
|
|
sort.Strings(funcStrings)
|
|
|
|
|
|
|
|
return fmt.Sprintf("%v", funcStrings)
|
|
|
|
}
|
|
|
|
|
2018-12-26 17:39:30 -08:00
|
|
|
var conditionFuncMap = map[name]func(Key, ValueSet) (Function, error){
|
|
|
|
stringEquals: newStringEqualsFunc,
|
|
|
|
stringNotEquals: newStringNotEqualsFunc,
|
|
|
|
stringEqualsIgnoreCase: newStringEqualsIgnoreCaseFunc,
|
|
|
|
stringNotEqualsIgnoreCase: newStringNotEqualsIgnoreCaseFunc,
|
|
|
|
binaryEquals: newBinaryEqualsFunc,
|
|
|
|
stringLike: newStringLikeFunc,
|
|
|
|
stringNotLike: newStringNotLikeFunc,
|
|
|
|
ipAddress: newIPAddressFunc,
|
|
|
|
notIPAddress: newNotIPAddressFunc,
|
|
|
|
null: newNullFunc,
|
2018-12-28 12:18:58 -08:00
|
|
|
boolean: newBooleanFunc,
|
2020-04-01 00:04:25 -07:00
|
|
|
numericEquals: newNumericEqualsFunc,
|
|
|
|
numericNotEquals: newNumericNotEqualsFunc,
|
|
|
|
numericLessThan: newNumericLessThanFunc,
|
|
|
|
numericLessThanEquals: newNumericLessThanEqualsFunc,
|
|
|
|
numericGreaterThan: newNumericGreaterThanFunc,
|
|
|
|
numericGreaterThanEquals: newNumericGreaterThanEqualsFunc,
|
|
|
|
dateEquals: newDateEqualsFunc,
|
|
|
|
dateNotEquals: newDateNotEqualsFunc,
|
|
|
|
dateLessThan: newDateLessThanFunc,
|
|
|
|
dateLessThanEquals: newDateLessThanEqualsFunc,
|
|
|
|
dateGreaterThan: newDateGreaterThanFunc,
|
|
|
|
dateGreaterThanEquals: newDateGreaterThanEqualsFunc,
|
2018-12-26 17:39:30 -08:00
|
|
|
// Add new conditions here.
|
|
|
|
}
|
|
|
|
|
2018-04-25 04:23:30 +05:30
|
|
|
// UnmarshalJSON - decodes JSON data to Functions.
|
|
|
|
func (functions *Functions) UnmarshalJSON(data []byte) error {
|
|
|
|
// As string kind, int kind then json.Unmarshaler is checked at
|
|
|
|
// https://github.com/golang/go/blob/master/src/encoding/json/decode.go#L618
|
|
|
|
// UnmarshalJSON() is not called for types extending string
|
|
|
|
// see https://play.golang.org/p/HrSsKksHvrS, better way to do is
|
|
|
|
// https://play.golang.org/p/y9ElWpBgVAB
|
|
|
|
//
|
|
|
|
// Due to this issue, name and Key types cannot be used as map keys below.
|
|
|
|
nm := make(map[string]map[string]ValueSet)
|
|
|
|
if err := json.Unmarshal(data, &nm); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(nm) == 0 {
|
|
|
|
return fmt.Errorf("condition must not be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
funcs := []Function{}
|
|
|
|
for nameString, args := range nm {
|
|
|
|
n, err := parseName(nameString)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for keyString, values := range args {
|
|
|
|
key, err := parseKey(keyString)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-12-26 17:39:30 -08:00
|
|
|
vfn, ok := conditionFuncMap[n]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("condition %v is not handled", n)
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := vfn(key, values)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2018-04-25 04:23:30 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
funcs = append(funcs, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*functions = funcs
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-07-09 11:18:48 +02:00
|
|
|
// GobEncode - encodes Functions to gob data.
|
|
|
|
func (functions Functions) GobEncode() ([]byte, error) {
|
|
|
|
return functions.MarshalJSON()
|
|
|
|
}
|
|
|
|
|
|
|
|
// GobDecode - decodes gob data to Functions.
|
|
|
|
func (functions *Functions) GobDecode(data []byte) error {
|
|
|
|
return functions.UnmarshalJSON(data)
|
|
|
|
}
|
|
|
|
|
2018-04-25 04:23:30 +05:30
|
|
|
// NewFunctions - returns new Functions with given function list.
|
|
|
|
func NewFunctions(functions ...Function) Functions {
|
|
|
|
return Functions(functions)
|
|
|
|
}
|