2017-02-07 22:33:23 +01:00
|
|
|
/*
|
|
|
|
Predicate package used to create interpreted mini languages with Go syntax - mostly to define
|
|
|
|
various predicates for configuration, e.g. Latency() > 40 || ErrorRate() > 0.5.
|
|
|
|
|
|
|
|
Here's an example of fully functional predicate language to deal with division remainders:
|
|
|
|
|
|
|
|
// takes number and returns true or false
|
|
|
|
type numberPredicate func(v int) bool
|
|
|
|
|
|
|
|
// Converts one number to another
|
|
|
|
type numberMapper func(v int) int
|
|
|
|
|
|
|
|
// Function that creates predicate to test if the remainder is 0
|
|
|
|
func divisibleBy(divisor int) numberPredicate {
|
|
|
|
return func(v int) bool {
|
|
|
|
return v%divisor == 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Function - logical operator AND that combines predicates
|
|
|
|
func numberAND(a, b numberPredicate) numberPredicate {
|
|
|
|
return func(v int) bool {
|
|
|
|
return a(v) && b(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := NewParser(Def{
|
|
|
|
Operators: Operators{
|
|
|
|
AND: numberAND,
|
|
|
|
},
|
|
|
|
Functions: map[string]interface{}{
|
|
|
|
"DivisibleBy": divisibleBy,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
pr, err := p.Parse("DivisibleBy(2) && DivisibleBy(3)")
|
|
|
|
if err == nil {
|
|
|
|
fmt.Fatalf("Error: %v", err)
|
|
|
|
}
|
|
|
|
pr.(numberPredicate)(2) // false
|
|
|
|
pr.(numberPredicate)(3) // false
|
|
|
|
pr.(numberPredicate)(6) // true
|
|
|
|
*/
|
|
|
|
package predicate
|
|
|
|
|
|
|
|
// Def contains supported operators (e.g. LT, GT) and functions passed in as a map.
|
|
|
|
type Def struct {
|
|
|
|
Operators Operators
|
|
|
|
// Function matching is case sensitive, e.g. Len is different from len
|
|
|
|
Functions map[string]interface{}
|
2018-01-22 12:16:03 +01:00
|
|
|
// GetIdentifier returns value of any identifier passed in
|
|
|
|
// in the form []string{"id", "field", "subfield"}
|
|
|
|
GetIdentifier GetIdentifierFn
|
|
|
|
// GetProperty returns property from a map
|
|
|
|
GetProperty GetPropertyFn
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 12:16:03 +01:00
|
|
|
// GetIdentifierFn function returns identifier based on selector
|
|
|
|
// e.g. id.field.subfield will be passed as.
|
|
|
|
// GetIdentifierFn([]string{"id", "field", "subfield"})
|
|
|
|
type GetIdentifierFn func(selector []string) (interface{}, error)
|
|
|
|
|
|
|
|
// GetPropertyFn reuturns property from a mapVal by key keyVal
|
|
|
|
type GetPropertyFn func(mapVal, keyVal interface{}) (interface{}, error)
|
|
|
|
|
2017-02-07 22:33:23 +01:00
|
|
|
// Operators contain functions for equality and logical comparison.
|
|
|
|
type Operators struct {
|
|
|
|
EQ interface{}
|
|
|
|
NEQ interface{}
|
|
|
|
|
|
|
|
LT interface{}
|
|
|
|
GT interface{}
|
|
|
|
|
|
|
|
LE interface{}
|
|
|
|
GE interface{}
|
|
|
|
|
|
|
|
OR interface{}
|
|
|
|
AND interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parser takes the string with expression and calls the operators and functions.
|
|
|
|
type Parser interface {
|
|
|
|
Parse(string) (interface{}, error)
|
|
|
|
}
|