Skip to content

Commit

Permalink
Merge pull request influxdata#470 from yosiat/eval-benchmarks
Browse files Browse the repository at this point in the history
Adding benchmark for stateful expression
  • Loading branch information
Nathaniel Cook committed Apr 22, 2016
2 parents b1b9310 + 729f110 commit 16d93a0
Showing 1 changed file with 292 additions and 0 deletions.
292 changes: 292 additions & 0 deletions tick/stateful_expr_benchmarks_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,292 @@
package tick_test

import (
"fmt"
"testing"

"github.com/influxdata/kapacitor/tick"
)

/*
Benchmarks for evaluating stateful expression
We will test by few basic scenario:
*) One operator
*) Two operators - with AND and OR relationship
And will test with reference node and without to check the
isolation performance overhead of reference node
Test name format:
Benchmark_{Evaluation type: EvalBool or EvalNum}_{Type: OneOperator, TwoOperator}_{LeftNode}_{RightNode}
*/

func Benchmark_EvalBool_OneOperator_NumberFloat64_NumberFloat64(b *testing.B) {

emptyScope := tick.NewScope()
benchmarkEvalBool(b, emptyScope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.NumberNode{
IsFloat: true,
Float64: float64(20),
},
Right: &tick.NumberNode{
IsFloat: true,
Float64: float64(10),
},
})
}

func Benchmark_EvalBool_OneOperator_NumberFloat64_NumberInt64(b *testing.B) {

emptyScope := tick.NewScope()
benchmarkEvalBool(b, emptyScope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.NumberNode{
IsFloat: true,
Float64: float64(20),
},
Right: &tick.NumberNode{
IsInt: true,
Int64: int64(10),
},
})
}

func Benchmark_EvalBool_OneOperator_NumberInt64_NumberInt64(b *testing.B) {
emptyScope := tick.NewScope()
benchmarkEvalBool(b, emptyScope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.NumberNode{
IsInt: true,
Int64: int64(20),
},
Right: &tick.NumberNode{
IsInt: true,
Int64: int64(10),
},
})
}

func Benchmark_EvalBool_OneOperator_ReferenceNodeFloat64_NumberFloat64(b *testing.B) {

scope := tick.NewScope()
scope.Set("value", float64(20))

benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsFloat: true,
Float64: float64(10),
},
})
}

func Benchmark_EvalBool_OneOperator_ReferenceNodeFloat64_NumberInt64(b *testing.B) {

scope := tick.NewScope()
scope.Set("value", float64(20))

benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsInt: true,
Int64: int64(10),
},
})
}

func Benchmark_EvalBool_OneOperator_ReferenceNodeFloat64_ReferenceNodeFloat64(b *testing.B) {

scope := tick.NewScope()
scope.Set("l", float64(20))
scope.Set("r", float64(10))
benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "l",
},
Right: &tick.ReferenceNode{
Reference: "r",
},
})
}

func Benchmark_EvalBool_OneOperatorWith11ScopeItem_ReferenceNodeFloat64_NumberFloat64(b *testing.B) {

scope := tick.NewScope()
scope.Set("value", float64(20))
for i := 0; i < 10; i++ {
scope.Set(fmt.Sprintf("value_%v", i), float64(i))
}

benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsFloat: true,
Float64: float64(10),
},
})
}

func Benchmark_EvalBool_OneOperatorValueChanges_ReferenceNodeFloat64_NumberFloat64(b *testing.B) {

scope := tick.NewScope()
initialValue := float64(20)

scope.Set("value", initialValue)

b.ReportAllocs()
b.ResetTimer()

se := tick.NewStatefulExpr(&tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsFloat: true,
Float64: float64(10),
},
})

// We have maximum value because we want to limit the maximum number in
// the reference node so we don't get too much big numbers and the benchmark suite will increase our iterations number (b.N)
currentValue := initialValue
maximumValue := float64(40)

var result bool
for i := 0; i < b.N; i++ {
b.StopTimer()

currentValue += float64(1)
if currentValue > maximumValue {
currentValue = initialValue
}

scope.Set("value", currentValue)

b.StartTimer()

result, err := se.EvalBool(scope)
if err != nil || !result {
v, _ := scope.Get("value")
b.Errorf("Failed to evaluate: error=%v, result=%t, value=%v, init=%v, maximum=%v", err, result, v, initialValue, maximumValue)
}
}

evalBoolResult = result
}

func Benchmark_EvalBool_OneOperator_ReferenceNodeInt64_ReferenceNodeInt64(b *testing.B) {

scope := tick.NewScope()
scope.Set("l", int64(20))
scope.Set("r", int64(10))

benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "l",
},
Right: &tick.ReferenceNode{
Reference: "r",
},
})
}

func Benchmark_EvalBool_OneOperatorWith11ScopeItem_ReferenceNodeInt64_NumberInt64(b *testing.B) {

scope := tick.NewScope()
scope.Set("value", int64(20))
for i := 0; i < 10; i++ {
scope.Set(fmt.Sprintf("value_%v", i), int64(i))
}

benchmarkEvalBool(b, scope, &tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsInt: true,
Int64: int64(10),
},
})
}

func Benchmark_EvalBool_OneOperatorValueChanges_ReferenceNodeInt64_NumberInt64(b *testing.B) {

scope := tick.NewScope()
initialValue := int64(20)

scope.Set("value", initialValue)

b.ReportAllocs()
b.ResetTimer()

se := tick.NewStatefulExpr(&tick.BinaryNode{
Operator: tick.TokenGreater,
Left: &tick.ReferenceNode{
Reference: "value",
},
Right: &tick.NumberNode{
IsInt: true,
Int64: int64(10),
},
})

// We have maximum value because we want to limit the maximum number in
// the reference node so we don't get too much big numbers and the benchmark suite will increase our iterations number (b.N)
currentValue := initialValue
maximumValue := int64(40)

var result bool
for i := 0; i < b.N; i++ {
b.StopTimer()

currentValue += int64(1)
if currentValue > maximumValue {
currentValue = initialValue
}

scope.Set("value", currentValue)

b.StartTimer()

result, err := se.EvalBool(scope)
if err != nil || !result {
v, _ := scope.Get("value")
b.Errorf("Failed to evaluate: error=%v, result=%t, value=%v, init=%v, maximum=%v", err, result, v, initialValue, maximumValue)
}
}

evalBoolResult = result
}

var evalBoolResult bool

func benchmarkEvalBool(b *testing.B, scope *tick.Scope, node tick.Node) {
b.ReportAllocs()
b.ResetTimer()

se := tick.NewStatefulExpr(node)
var err error

for i := 0; i < b.N; i++ {
evalBoolResult, err = se.EvalBool(scope)
if err != nil {
b.Fatal(err)
}
}

}

0 comments on commit 16d93a0

Please sign in to comment.