package govaluate import ( "errors" "fmt" "regexp" "testing" "time" ) /* Represents a test of expression evaluation */ type EvaluationTest struct { Name string Input string Functions map[string]ExpressionFunction Parameters []EvaluationParameter Expected interface{} } type EvaluationParameter struct { Name string Value interface{} } func TestNoParameterEvaluation(test *testing.T) { evaluationTests := []EvaluationTest{ EvaluationTest{ Name: "Single PLUS", Input: "51 + 49", Expected: 100.0, }, EvaluationTest{ Name: "Single MINUS", Input: "100 - 51", Expected: 49.0, }, EvaluationTest{ Name: "Single BITWISE AND", Input: "100 & 50", Expected: 32.0, }, EvaluationTest{ Name: "Single BITWISE OR", Input: "100 | 50", Expected: 118.0, }, EvaluationTest{ Name: "Single BITWISE XOR", Input: "100 ^ 50", Expected: 86.0, }, EvaluationTest{ Name: "Single shift left", Input: "2 << 1", Expected: 4.0, }, EvaluationTest{ Name: "Single shift right", Input: "2 >> 1", Expected: 1.0, }, EvaluationTest{ Name: "Single BITWISE NOT", Input: "~10", Expected: -11.0, }, EvaluationTest{ Name: "Single MULTIPLY", Input: "5 * 20", Expected: 100.0, }, EvaluationTest{ Name: "Single DIVIDE", Input: "100 / 20", Expected: 5.0, }, EvaluationTest{ Name: "Single even MODULUS", Input: "100 % 2", Expected: 0.0, }, EvaluationTest{ Name: "Single odd MODULUS", Input: "101 % 2", Expected: 1.0, }, EvaluationTest{ Name: "Single EXPONENT", Input: "10 ** 2", Expected: 100.0, }, EvaluationTest{ Name: "Compound PLUS", Input: "20 + 30 + 50", Expected: 100.0, }, EvaluationTest{ Name: "Compound BITWISE AND", Input: "20 & 30 & 50", Expected: 16.0, }, EvaluationTest{ Name: "Mutiple operators", Input: "20 * 5 - 49", Expected: 51.0, }, EvaluationTest{ Name: "Parenthesis usage", Input: "100 - (5 * 10)", Expected: 50.0, }, EvaluationTest{ Name: "Nested parentheses", Input: "50 + (5 * (15 - 5))", Expected: 100.0, }, EvaluationTest{ Name: "Nested parentheses with bitwise", Input: "100 ^ (23 * (2 | 5))", Expected: 197.0, }, EvaluationTest{ Name: "Logical OR operation of two clauses", Input: "(1 == 1) || (true == true)", Expected: true, }, EvaluationTest{ Name: "Logical AND operation of two clauses", Input: "(1 == 1) && (true == true)", Expected: true, }, EvaluationTest{ Name: "Implicit boolean", Input: "2 > 1", Expected: true, }, EvaluationTest{ Name: "Compound boolean", Input: "5 < 10 && 1 < 5", Expected: true, }, EvaluationTest{ Name: "Evaluated true && false operation (for issue #8)", Input: "1 > 10 && 11 > 10", Expected: false, }, EvaluationTest{ Name: "Evaluated true && false operation (for issue #8)", Input: "true == true && false == true", Expected: false, }, EvaluationTest{ Name: "Parenthesis boolean", Input: "10 < 50 && (1 != 2 && 1 > 0)", Expected: true, }, EvaluationTest{ Name: "Comparison of string constants", Input: "'foo' == 'foo'", Expected: true, }, EvaluationTest{ Name: "NEQ comparison of string constants", Input: "'foo' != 'bar'", Expected: true, }, EvaluationTest{ Name: "REQ comparison of string constants", Input: "'foobar' =~ 'oba'", Expected: true, }, EvaluationTest{ Name: "NREQ comparison of string constants", Input: "'foo' !~ 'bar'", Expected: true, }, EvaluationTest{ Name: "Multiplicative/additive order", Input: "5 + 10 * 2", Expected: 25.0, }, EvaluationTest{ Name: "Multiple constant multiplications", Input: "10 * 10 * 10", Expected: 1000.0, }, EvaluationTest{ Name: "Multiple adds/multiplications", Input: "10 * 10 * 10 + 1 * 10 * 10", Expected: 1100.0, }, EvaluationTest{ Name: "Modulus precedence", Input: "1 + 101 % 2 * 5", Expected: 6.0, }, EvaluationTest{ Name: "Exponent precedence", Input: "1 + 5 ** 3 % 2 * 5", Expected: 6.0, }, EvaluationTest{ Name: "Bit shift precedence", Input: "50 << 1 & 90", Expected: 64.0, }, EvaluationTest{ Name: "Bit shift precedence", Input: "90 & 50 << 1", Expected: 64.0, }, EvaluationTest{ Name: "Bit shift precedence amongst non-bitwise", Input: "90 + 50 << 1 * 5", Expected: 4480.0, }, EvaluationTest{ Name: "Order of non-commutative same-precedence operators (additive)", Input: "1 - 2 - 4 - 8", Expected: -13.0, }, EvaluationTest{ Name: "Order of non-commutative same-precedence operators (multiplicative)", Input: "1 * 4 / 2 * 8", Expected: 16.0, }, EvaluationTest{ Name: "Null coalesce precedence", Input: "true ?? true ? 100 + 200 : 400", Expected: 300.0, }, EvaluationTest{ Name: "Identical date equivalence", Input: "'2014-01-02 14:12:22' == '2014-01-02 14:12:22'", Expected: true, }, EvaluationTest{ Name: "Positive date GT", Input: "'2014-01-02 14:12:22' > '2014-01-02 12:12:22'", Expected: true, }, EvaluationTest{ Name: "Negative date GT", Input: "'2014-01-02 14:12:22' > '2014-01-02 16:12:22'", Expected: false, }, EvaluationTest{ Name: "Positive date GTE", Input: "'2014-01-02 14:12:22' >= '2014-01-02 12:12:22'", Expected: true, }, EvaluationTest{ Name: "Negative date GTE", Input: "'2014-01-02 14:12:22' >= '2014-01-02 16:12:22'", Expected: false, }, EvaluationTest{ Name: "Positive date LT", Input: "'2014-01-02 14:12:22' < '2014-01-02 16:12:22'", Expected: true, }, EvaluationTest{ Name: "Negative date LT", Input: "'2014-01-02 14:12:22' < '2014-01-02 11:12:22'", Expected: false, }, EvaluationTest{ Name: "Positive date LTE", Input: "'2014-01-02 09:12:22' <= '2014-01-02 12:12:22'", Expected: true, }, EvaluationTest{ Name: "Negative date LTE", Input: "'2014-01-02 14:12:22' <= '2014-01-02 11:12:22'", Expected: false, }, EvaluationTest{ Name: "Sign prefix comparison", Input: "-1 < 0", Expected: true, }, EvaluationTest{ Name: "Lexicographic LT", Input: "'ab' < 'abc'", Expected: true, }, EvaluationTest{ Name: "Lexicographic LTE", Input: "'ab' <= 'abc'", Expected: true, }, EvaluationTest{ Name: "Lexicographic GT", Input: "'aba' > 'abc'", Expected: false, }, EvaluationTest{ Name: "Lexicographic GTE", Input: "'aba' >= 'abc'", Expected: false, }, EvaluationTest{ Name: "Boolean sign prefix comparison", Input: "!true == false", Expected: true, }, EvaluationTest{ Name: "Inversion of clause", Input: "!(10 < 0)", Expected: true, }, EvaluationTest{ Name: "Negation after modifier", Input: "10 * -10", Expected: -100.0, }, EvaluationTest{ Name: "Ternary with single boolean", Input: "true ? 10", Expected: 10.0, }, EvaluationTest{ Name: "Ternary nil with single boolean", Input: "false ? 10", Expected: nil, }, EvaluationTest{ Name: "Ternary with comparator boolean", Input: "10 > 5 ? 35.50", Expected: 35.50, }, EvaluationTest{ Name: "Ternary nil with comparator boolean", Input: "1 > 5 ? 35.50", Expected: nil, }, EvaluationTest{ Name: "Ternary with parentheses", Input: "(5 * (15 - 5)) > 5 ? 35.50", Expected: 35.50, }, EvaluationTest{ Name: "Ternary precedence", Input: "true ? 35.50 > 10", Expected: true, }, EvaluationTest{ Name: "Ternary-else", Input: "false ? 35.50 : 50", Expected: 50.0, }, EvaluationTest{ Name: "Ternary-else inside clause", Input: "(false ? 5 : 35.50) > 10", Expected: true, }, EvaluationTest{ Name: "Ternary-else (true-case) inside clause", Input: "(true ? 1 : 5) < 10", Expected: true, }, EvaluationTest{ Name: "Ternary-else before comparator (negative case)", Input: "true ? 1 : 5 > 10", Expected: 1.0, }, EvaluationTest{ Name: "Nested ternaries (#32)", Input: "(2 == 2) ? 1 : (true ? 2 : 3)", Expected: 1.0, }, EvaluationTest{ Name: "Nested ternaries, right case (#32)", Input: "false ? 1 : (true ? 2 : 3)", Expected: 2.0, }, EvaluationTest{ Name: "Doubly-nested ternaries (#32)", Input: "true ? (false ? 1 : (false ? 2 : 3)) : (false ? 4 : 5)", Expected: 3.0, }, EvaluationTest{ Name: "String to string concat", Input: "'foo' + 'bar' == 'foobar'", Expected: true, }, EvaluationTest{ Name: "String to float64 concat", Input: "'foo' + 123 == 'foo123'", Expected: true, }, EvaluationTest{ Name: "Float64 to string concat", Input: "123 + 'bar' == '123bar'", Expected: true, }, EvaluationTest{ Name: "String to date concat", Input: "'foo' + '02/05/1970' == 'foobar'", Expected: false, }, EvaluationTest{ Name: "String to bool concat", Input: "'foo' + true == 'footrue'", Expected: true, }, EvaluationTest{ Name: "Bool to string concat", Input: "true + 'bar' == 'truebar'", Expected: true, }, EvaluationTest{ Name: "Null coalesce left", Input: "1 ?? 2", Expected: 1.0, }, EvaluationTest{ Name: "Array membership literals", Input: "1 in (1, 2, 3)", Expected: true, }, EvaluationTest{ Name: "Array membership literal with inversion", Input: "!(1 in (1, 2, 3))", Expected: false, }, EvaluationTest{ Name: "Logical operator reordering (#30)", Input: "(true && true) || (true && false)", Expected: true, }, EvaluationTest{ Name: "Logical operator reordering without parens (#30)", Input: "true && true || true && false", Expected: true, }, EvaluationTest{ Name: "Logical operator reordering with multiple OR (#30)", Input: "false || true && true || false", Expected: true, }, EvaluationTest{ Name: "Left-side multiple consecutive (should be reordered) operators", Input: "(10 * 10 * 10) > 10", Expected: true, }, EvaluationTest{ Name: "Three-part non-paren logical op reordering (#44)", Input: "false && true || true", Expected: true, }, EvaluationTest{ Name: "Three-part non-paren logical op reordering (#44), second one", Input: "true || false && true", Expected: true, }, EvaluationTest{ Name: "Logical operator reordering without parens (#45)", Input: "true && true || false && false", Expected: true, }, EvaluationTest{ Name: "Single function", Input: "foo()", Functions: map[string]ExpressionFunction{ "foo": func(arguments ...interface{}) (interface{}, error) { return true, nil }, }, Expected: true, }, EvaluationTest{ Name: "Function with argument", Input: "passthrough(1)", Functions: map[string]ExpressionFunction{ "passthrough": func(arguments ...interface{}) (interface{}, error) { return arguments[0], nil }, }, Expected: 1.0, }, EvaluationTest{ Name: "Function with arguments", Input: "passthrough(1, 2)", Functions: map[string]ExpressionFunction{ "passthrough": func(arguments ...interface{}) (interface{}, error) { return arguments[0].(float64) + arguments[1].(float64), nil }, }, Expected: 3.0, }, EvaluationTest{ Name: "Nested function with precedence", Input: "sum(1, sum(2, 3), 2 + 2, true ? 4 : 5)", Functions: map[string]ExpressionFunction{ "sum": func(arguments ...interface{}) (interface{}, error) { sum := 0.0 for _, v := range arguments { sum += v.(float64) } return sum, nil }, }, Expected: 14.0, }, EvaluationTest{ Name: "Empty function and modifier, compared", Input: "numeric()-1 > 0", Functions: map[string]ExpressionFunction{ "numeric": func(arguments ...interface{}) (interface{}, error) { return 2.0, nil }, }, Expected: true, }, EvaluationTest{ Name: "Empty function comparator", Input: "numeric() > 0", Functions: map[string]ExpressionFunction{ "numeric": func(arguments ...interface{}) (interface{}, error) { return 2.0, nil }, }, Expected: true, }, EvaluationTest{ Name: "Empty function logical operator", Input: "success() && !false", Functions: map[string]ExpressionFunction{ "success": func(arguments ...interface{}) (interface{}, error) { return true, nil }, }, Expected: true, }, EvaluationTest{ Name: "Empty function ternary", Input: "nope() ? 1 : 2.0", Functions: map[string]ExpressionFunction{ "nope": func(arguments ...interface{}) (interface{}, error) { return false, nil }, }, Expected: 2.0, }, EvaluationTest{ Name: "Empty function null coalesce", Input: "null() ?? 2", Functions: map[string]ExpressionFunction{ "null": func(arguments ...interface{}) (interface{}, error) { return nil, nil }, }, Expected: 2.0, }, EvaluationTest{ Name: "Empty function with prefix", Input: "-ten()", Functions: map[string]ExpressionFunction{ "ten": func(arguments ...interface{}) (interface{}, error) { return 10.0, nil }, }, Expected: -10.0, }, EvaluationTest{ Name: "Empty function as part of chain", Input: "10 - numeric() - 2", Functions: map[string]ExpressionFunction{ "numeric": func(arguments ...interface{}) (interface{}, error) { return 5.0, nil }, }, Expected: 3.0, }, EvaluationTest{ Name: "Empty function near separator", Input: "10 in (1, 2, 3, ten(), 8)", Functions: map[string]ExpressionFunction{ "ten": func(arguments ...interface{}) (interface{}, error) { return 10.0, nil }, }, Expected: true, }, EvaluationTest{ Name: "Enclosed empty function with modifier and comparator (#28)", Input: "(ten() - 1) > 3", Functions: map[string]ExpressionFunction{ "ten": func(arguments ...interface{}) (interface{}, error) { return 10.0, nil }, }, Expected: true, }, EvaluationTest{ Name: "Ternary/Java EL ambiguity", Input: "false ? foo:length()", Functions: map[string]ExpressionFunction{ "length": func(arguments ...interface{}) (interface{}, error) { return 1.0, nil }, }, Expected: 1.0, }, } runEvaluationTests(evaluationTests, test) } func TestParameterizedEvaluation(test *testing.T) { evaluationTests := []EvaluationTest{ EvaluationTest{ Name: "Single parameter modified by constant", Input: "foo + 2", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 2.0, }, }, Expected: 4.0, }, EvaluationTest{ Name: "Single parameter modified by variable", Input: "foo * bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 5.0, }, EvaluationParameter{ Name: "bar", Value: 2.0, }, }, Expected: 10.0, }, EvaluationTest{ Name: "Multiple multiplications of the same parameter", Input: "foo * foo * foo", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 10.0, }, }, Expected: 1000.0, }, EvaluationTest{ Name: "Multiple additions of the same parameter", Input: "foo + foo + foo", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 10.0, }, }, Expected: 30.0, }, EvaluationTest{ Name: "Parameter name sensitivity", Input: "foo + FoO + FOO", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 8.0, }, EvaluationParameter{ Name: "FoO", Value: 4.0, }, EvaluationParameter{ Name: "FOO", Value: 2.0, }, }, Expected: 14.0, }, EvaluationTest{ Name: "Sign prefix comparison against prefixed variable", Input: "-1 < -foo", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: -8.0, }, }, Expected: true, }, EvaluationTest{ Name: "Fixed-point parameter", Input: "foo > 1", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 2, }, }, Expected: true, }, EvaluationTest{ Name: "Modifier after closing clause", Input: "(2 + 2) + 2 == 6", Expected: true, }, EvaluationTest{ Name: "Comparator after closing clause", Input: "(2 + 2) >= 4", Expected: true, }, EvaluationTest{ Name: "Two-boolean logical operation (for issue #8)", Input: "(foo == true) || (bar == true)", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: true, }, EvaluationParameter{ Name: "bar", Value: false, }, }, Expected: true, }, EvaluationTest{ Name: "Two-variable integer logical operation (for issue #8)", Input: "foo > 10 && bar > 10", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 1, }, EvaluationParameter{ Name: "bar", Value: 11, }, }, Expected: false, }, EvaluationTest{ Name: "Regex against right-hand parameter", Input: "'foobar' =~ foo", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "obar", }, }, Expected: true, }, EvaluationTest{ Name: "Not-regex against right-hand parameter", Input: "'foobar' !~ foo", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "baz", }, }, Expected: true, }, EvaluationTest{ Name: "Regex against two parameters", Input: "foo =~ bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "foobar", }, EvaluationParameter{ Name: "bar", Value: "oba", }, }, Expected: true, }, EvaluationTest{ Name: "Not-regex against two parameters", Input: "foo !~ bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "foobar", }, EvaluationParameter{ Name: "bar", Value: "baz", }, }, Expected: true, }, EvaluationTest{ Name: "Pre-compiled regex", Input: "foo =~ bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "foobar", }, EvaluationParameter{ Name: "bar", Value: regexp.MustCompile("[fF][oO]+"), }, }, Expected: true, }, EvaluationTest{ Name: "Pre-compiled not-regex", Input: "foo !~ bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "foobar", }, EvaluationParameter{ Name: "bar", Value: regexp.MustCompile("[fF][oO]+"), }, }, Expected: false, }, EvaluationTest{ Name: "Single boolean parameter", Input: "commission ? 10", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "commission", Value: true, }, }, Expected: 10.0, }, EvaluationTest{ Name: "True comparator with a parameter", Input: "partner == 'amazon' ? 10", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "partner", Value: "amazon", }, }, Expected: 10.0, }, EvaluationTest{ Name: "False comparator with a parameter", Input: "partner == 'amazon' ? 10", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "partner", Value: "ebay", }, }, Expected: nil, }, EvaluationTest{ Name: "True comparator with multiple parameters", Input: "theft && period == 24 ? 60", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "theft", Value: true, }, EvaluationParameter{ Name: "period", Value: 24, }, }, Expected: 60.0, }, EvaluationTest{ Name: "False comparator with multiple parameters", Input: "theft && period == 24 ? 60", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "theft", Value: false, }, EvaluationParameter{ Name: "period", Value: 24, }, }, Expected: nil, }, EvaluationTest{ Name: "String concat with single string parameter", Input: "foo + 'bar'", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "baz", }, }, Expected: "bazbar", }, EvaluationTest{ Name: "String concat with multiple string parameter", Input: "foo + bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "baz", }, EvaluationParameter{ Name: "bar", Value: "quux", }, }, Expected: "bazquux", }, EvaluationTest{ Name: "String concat with float parameter", Input: "foo + bar", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "baz", }, EvaluationParameter{ Name: "bar", Value: 123.0, }, }, Expected: "baz123", }, EvaluationTest{ Name: "Mixed multiple string concat", Input: "foo + 123 + 'bar' + true", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: "baz", }, }, Expected: "baz123bartrue", }, EvaluationTest{ Name: "Integer width spectrum", Input: "uint8 + uint16 + uint32 + uint64 + int8 + int16 + int32 + int64", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "uint8", Value: uint8(0), }, EvaluationParameter{ Name: "uint16", Value: uint16(0), }, EvaluationParameter{ Name: "uint32", Value: uint32(0), }, EvaluationParameter{ Name: "uint64", Value: uint64(0), }, EvaluationParameter{ Name: "int8", Value: int8(0), }, EvaluationParameter{ Name: "int16", Value: int16(0), }, EvaluationParameter{ Name: "int32", Value: int32(0), }, EvaluationParameter{ Name: "int64", Value: int64(0), }, }, Expected: 0.0, }, EvaluationTest{ Name: "Floats", Input: "float32 + float64", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "float32", Value: float32(0.0), }, EvaluationParameter{ Name: "float64", Value: float64(0.0), }, }, Expected: 0.0, }, EvaluationTest{ Name: "Null coalesce right", Input: "foo ?? 1.0", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: nil, }, }, Expected: 1.0, }, EvaluationTest{ Name: "Multiple comparator/logical operators (#30)", Input: "(foo >= 2887057408 && foo <= 2887122943) || (foo >= 168100864 && foo <= 168118271)", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 2887057409, }, }, Expected: true, }, EvaluationTest{ Name: "Multiple comparator/logical operators, opposite order (#30)", Input: "(foo >= 168100864 && foo <= 168118271) || (foo >= 2887057408 && foo <= 2887122943)", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 2887057409, }, }, Expected: true, }, EvaluationTest{ Name: "Multiple comparator/logical operators, small value (#30)", Input: "(foo >= 2887057408 && foo <= 2887122943) || (foo >= 168100864 && foo <= 168118271)", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 168100865, }, }, Expected: true, }, EvaluationTest{ Name: "Multiple comparator/logical operators, small value, opposite order (#30)", Input: "(foo >= 168100864 && foo <= 168118271) || (foo >= 2887057408 && foo <= 2887122943)", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "foo", Value: 168100865, }, }, Expected: true, }, EvaluationTest{ Name: "Incomparable array equality comparison", Input: "arr == arr", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "arr", Value: []int{0, 0, 0}, }, }, Expected: true, }, EvaluationTest{ Name: "Incomparable array not-equality comparison", Input: "arr != arr", Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "arr", Value: []int{0, 0, 0}, }, }, Expected: false, }, EvaluationTest{ Name: "Mixed function and parameters", Input: "sum(1.2, amount) + name", Functions: map[string]ExpressionFunction{ "sum": func(arguments ...interface{}) (interface{}, error) { sum := 0.0 for _, v := range arguments { sum += v.(float64) } return sum, nil }, }, Parameters: []EvaluationParameter{ EvaluationParameter{ Name: "amount", Value: .8, }, EvaluationParameter{ Name: "name", Value: "awesome", }, }, Expected: "2awesome", }, EvaluationTest{ Name: "Short-circuit OR", Input: "true || fail()", Functions: map[string]ExpressionFunction{ "fail": func(arguments ...interface{}) (interface{}, error) { return nil, errors.New("Did not short-circuit") }, }, Expected: true, }, EvaluationTest{ Name: "Short-circuit AND", Input: "false && fail()", Functions: map[string]ExpressionFunction{ "fail": func(arguments ...interface{}) (interface{}, error) { return nil, errors.New("Did not short-circuit") }, }, Expected: false, }, EvaluationTest{ Name: "Short-circuit ternary", Input: "true ? 1 : fail()", Functions: map[string]ExpressionFunction{ "fail": func(arguments ...interface{}) (interface{}, error) { return nil, errors.New("Did not short-circuit") }, }, Expected: 1.0, }, EvaluationTest{ Name: "Short-circuit coalesce", Input: "'foo' ?? fail()", Functions: map[string]ExpressionFunction{ "fail": func(arguments ...interface{}) (interface{}, error) { return nil, errors.New("Did not short-circuit") }, }, Expected: "foo", }, EvaluationTest{ Name: "Simple parameter call", Input: "foo.String", Parameters: []EvaluationParameter{fooParameter}, Expected: fooParameter.Value.(dummyParameter).String, }, EvaluationTest{ Name: "Simple parameter function call", Input: "foo.Func()", Parameters: []EvaluationParameter{fooParameter}, Expected: "funk", }, EvaluationTest{ Name: "Simple parameter call from pointer", Input: "fooptr.String", Parameters: []EvaluationParameter{fooPtrParameter}, Expected: fooParameter.Value.(dummyParameter).String, }, EvaluationTest{ Name: "Simple parameter function call from pointer", Input: "fooptr.Func()", Parameters: []EvaluationParameter{fooPtrParameter}, Expected: "funk", }, EvaluationTest{ Name: "Simple parameter function call from pointer", Input: "fooptr.Func3()", Parameters: []EvaluationParameter{fooPtrParameter}, Expected: "fronk", }, EvaluationTest{ Name: "Simple parameter call", Input: "foo.String == 'hi'", Parameters: []EvaluationParameter{fooParameter}, Expected: false, }, EvaluationTest{ Name: "Simple parameter call with modifier", Input: "foo.String + 'hi'", Parameters: []EvaluationParameter{fooParameter}, Expected: fooParameter.Value.(dummyParameter).String + "hi", }, EvaluationTest{ Name: "Simple parameter function call, two-arg return", Input: "foo.Func2()", Parameters: []EvaluationParameter{fooParameter}, Expected: "frink", }, EvaluationTest{ Name: "Parameter function call with all argument types", Input: "foo.TestArgs(\"hello\", 1, 2, 3, 4, 5, 1, 2, 3, 4, 5, 1.0, 2.0, true)", Parameters: []EvaluationParameter{fooParameter}, Expected: "hello: 33", }, EvaluationTest{ Name: "Simple parameter function call, one arg", Input: "foo.FuncArgStr('boop')", Parameters: []EvaluationParameter{fooParameter}, Expected: "boop", }, EvaluationTest{ Name: "Simple parameter function call, one arg", Input: "foo.FuncArgStr('boop') + 'hi'", Parameters: []EvaluationParameter{fooParameter}, Expected: "boophi", }, EvaluationTest{ Name: "Nested parameter function call", Input: "foo.Nested.Dunk('boop')", Parameters: []EvaluationParameter{fooParameter}, Expected: "boopdunk", }, EvaluationTest{ Name: "Nested parameter call", Input: "foo.Nested.Funk", Parameters: []EvaluationParameter{fooParameter}, Expected: "funkalicious", }, EvaluationTest{ Name: "Parameter call with + modifier", Input: "1 + foo.Int", Parameters: []EvaluationParameter{fooParameter}, Expected: 102.0, }, EvaluationTest{ Name: "Parameter string call with + modifier", Input: "'woop' + (foo.String)", Parameters: []EvaluationParameter{fooParameter}, Expected: "woopstring!", }, EvaluationTest{ Name: "Parameter call with && operator", Input: "true && foo.BoolFalse", Parameters: []EvaluationParameter{fooParameter}, Expected: false, }, EvaluationTest{ Name: "Null coalesce nested parameter", Input: "foo.Nil ?? false", Parameters: []EvaluationParameter{fooParameter}, Expected: false, }, } runEvaluationTests(evaluationTests, test) } /* Tests the behavior of a nil set of parameters. */ func TestNilParameters(test *testing.T) { expression, _ := NewEvaluableExpression("true") _, err := expression.Evaluate(nil) if err != nil { test.Fail() } } /* Tests functionality related to using functions with a struct method receiver. Created to test #54. */ func TestStructFunctions(test *testing.T) { parseFormat := "2006" y2k, _ := time.Parse(parseFormat, "2000") y2k1, _ := time.Parse(parseFormat, "2001") functions := map[string]ExpressionFunction{ "func1": func(args ...interface{}) (interface{}, error) { return float64(y2k.Year()), nil }, "func2": func(args ...interface{}) (interface{}, error) { return float64(y2k1.Year()), nil }, } exp, _ := NewEvaluableExpressionWithFunctions("func1() + func2()", functions) result, _ := exp.Evaluate(nil) if result != 4001.0 { test.Logf("Function calling method did not return the right value. Got: %v, expected %d\n", result, 4001) test.Fail() } } func runEvaluationTests(evaluationTests []EvaluationTest, test *testing.T) { var expression *EvaluableExpression var result interface{} var parameters map[string]interface{} var err error fmt.Printf("Running %d evaluation test cases...\n", len(evaluationTests)) // Run the test cases. for _, evaluationTest := range evaluationTests { if evaluationTest.Functions != nil { expression, err = NewEvaluableExpressionWithFunctions(evaluationTest.Input, evaluationTest.Functions) } else { expression, err = NewEvaluableExpression(evaluationTest.Input) } if err != nil { test.Logf("Test '%s' failed to parse: '%s'", evaluationTest.Name, err) test.Fail() continue } parameters = make(map[string]interface{}, 8) for _, parameter := range evaluationTest.Parameters { parameters[parameter.Name] = parameter.Value } result, err = expression.Evaluate(parameters) if err != nil { test.Logf("Test '%s' failed", evaluationTest.Name) test.Logf("Encountered error: %s", err.Error()) test.Fail() continue } if result != evaluationTest.Expected { test.Logf("Test '%s' failed", evaluationTest.Name) test.Logf("Evaluation result '%v' does not match expected: '%v'", result, evaluationTest.Expected) test.Fail() } } }