Skip to content

Commit

Permalink
fix(i): Flaky normalize filter test (sourcenetwork#1912)
Browse files Browse the repository at this point in the history
## Relevant issue(s)

Resolves sourcenetwork#1879 

## Description

This PR attempts to fix a flaky filter test. The normalization logic has
been refactored to a recursive / functional approach that (hopefully)
makes it a little easier to understand.

## Tasks

- [x] I made sure the code is well commented, particularly
hard-to-understand areas.
- [x] I made sure the repository-held documentation is changed
accordingly.
- [x] I made sure the pull request title adheres to the conventional
commit style (the subset used in the project can be found in
[tools/configs/chglog/config.yml](tools/configs/chglog/config.yml)).
- [x] I made sure to discuss its limitations such as threats to
validity, vulnerability to mistake and misuse, robustness to
invalidation of assumptions, resource requirements, ...

## How has this been tested?

Ran normalize tests in a loop for 200k+ iterations.

Specify the platform(s) on which this was tested:
- MacOS
  • Loading branch information
nasdf authored Sep 27, 2023
1 parent 7ab41f4 commit 0ca46a0
Showing 1 changed file with 173 additions and 102 deletions.
275 changes: 173 additions & 102 deletions planner/filter/normalize.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,134 +16,205 @@ import (
)

// normalize normalizes the provided filter conditions.
//
// The following cases are subject of normalization:
// - _and or _or with one element is removed flattened
// - double _not is removed
// - any number of consecutive _ands with any number of elements is flattened
//
// As the result object is a map with unique keys (a.k.a. properties),
// while performing flattening of compound operators if the same property
// is present in the result map, both conditions will be moved into an _and
func normalize(conditions map[connor.FilterKey]any) map[connor.FilterKey]any {
return normalizeConditions(conditions, false).(map[connor.FilterKey]any)
return normalizeCondition(nil, conditions).(map[connor.FilterKey]any)
}

func conditionsArrToMap(conditions []any) map[connor.FilterKey]any {
// normalizeCondition returns a normalized version of the given condition.
func normalizeCondition(parentKey connor.FilterKey, condition any) (result any) {
switch t := condition.(type) {
case map[connor.FilterKey]any:
result = normalizeConditions(parentKey, t)

case []any:
conditions := make([]any, len(t))
for i, c := range t {
conditions[i] = normalizeCondition(parentKey, c)
}
result = conditions

default:
result = t
}

return normalizeProperty(parentKey, result)
}

// normalizeConditions returns a normalized version of the given conditions.
func normalizeConditions(parentKey connor.FilterKey, conditions map[connor.FilterKey]any) map[connor.FilterKey]any {
result := make(map[connor.FilterKey]any)
for _, clause := range conditions {
if clauseMap, ok := clause.(map[connor.FilterKey]any); ok {
for k, v := range clauseMap {
result[k] = v
for key, val := range conditions {
result[key] = normalizeCondition(key, val)

// check if the condition is an operator that can be normalized
op, ok := key.(*mapper.Operator)
if !ok {
continue
}
// check if we have any conditions that can be merged
merge := normalizeOperator(parentKey, op, result[key])
if len(merge) == 0 {
continue
}
delete(result, key)

// merge properties directly into result
for _, c := range merge {
for key, val := range c.(map[connor.FilterKey]any) {
result[key] = val
}
}

// if the merged filter was an _or operator
// there may be child filters that can be merged
if op.Operation == request.FilterOpOr {
result = normalizeConditions(parentKey, result)
}
}
return result
}

func addNormalizedCondition(key connor.FilterKey, val any, m map[connor.FilterKey]any) {
if _, isProp := key.(*mapper.PropertyIndex); isProp {
var andOp *mapper.Operator
var andContent []any
for existingKey := range m {
if op, isOp := existingKey.(*mapper.Operator); isOp && op.Operation == request.FilterOpAnd {
andOp = op
andContent = m[existingKey].([]any)
break
}
}
for existingKey := range m {
if existingKey.Equal(key) {
existingVal := m[existingKey]
delete(m, existingKey)
if andOp == nil {
andOp = &mapper.Operator{Operation: request.FilterOpAnd}
}
m[andOp] = append(
andContent,
map[connor.FilterKey]any{existingKey: existingVal},
map[connor.FilterKey]any{key: val},
)
return
}
// normalizeOperator returns a normalized array of conditions.
func normalizeOperator(parentKey connor.FilterKey, op *mapper.Operator, condition any) []any {
switch op.Operation {
case request.FilterOpNot:
return normalizeOperatorNot(condition)

case request.FilterOpOr:
return normalizeOperatorOr(condition)

case request.FilterOpAnd:
return normalizeOperatorAnd(parentKey, condition)

default:
return nil
}
}

// normalizeOperatorAnd returns an array of conditions with all _and operators merged.
//
// If the parent operator is _not or _or, the subconditions will not be merged.
func normalizeOperatorAnd(parentKey connor.FilterKey, condition any) []any {
result := condition.([]any)
// always merge if only 1 property
if len(result) == 1 {
return result
}
// always merge if parent is not an operator
parentOp, ok := parentKey.(*mapper.Operator)
if !ok {
return result
}
// don't merge if parent is a _not or _or operator
if parentOp.Operation == request.FilterOpNot || parentOp.Operation == request.FilterOpOr {
return nil
}
return result
}

// normalizeOperatorOr returns an array of conditions with all single _or operators merged.
func normalizeOperatorOr(condition any) []any {
result := condition.([]any)
// don't merge if more than 1 property
if len(result) > 1 {
return nil
}
return result
}

// normalizeOperatorNot returns an array of conditions with all double _not operators merged.
func normalizeOperatorNot(condition any) (result []any) {
subConditions := condition.(map[connor.FilterKey]any)
// don't merge if more than 1 property
if len(subConditions) > 1 {
return nil
}
// find double _not occurances
for subKey, subCondition := range subConditions {
op, ok := subKey.(*mapper.Operator)
if ok && op.Operation == request.FilterOpNot {
result = append(result, subCondition)
}
for _, andElement := range andContent {
elementMap := andElement.(map[connor.FilterKey]any)
for andElementKey := range elementMap {
if andElementKey.Equal(key) {
m[andOp] = append(andContent, map[connor.FilterKey]any{key: val})
return
}
}
return result
}

// normalizeProperty flattens and groups property filters where possible.
//
// Filters targeting the same property will be grouped into a single _and.
func normalizeProperty(parentKey connor.FilterKey, condition any) any {
switch t := condition.(type) {
case map[connor.FilterKey]any:
results := make(map[connor.FilterKey]any)
for _, c := range normalizeProperties(parentKey, []any{t}) {
for key, val := range c.(map[connor.FilterKey]any) {
results[key] = val
}
}
return results

case []any:
return normalizeProperties(parentKey, t)

default:
return t
}
m[key] = val
}

func normalizeConditions(conditions any, skipRoot bool) any {
result := make(map[connor.FilterKey]any)
switch typedConditions := conditions.(type) {
case map[connor.FilterKey]any:
for rootKey, rootVal := range typedConditions {
rootOpKey, isRootOp := rootKey.(*mapper.Operator)
if isRootOp {
if rootOpKey.Operation == request.FilterOpAnd || rootOpKey.Operation == request.FilterOpOr {
rootValArr := rootVal.([]any)
if len(rootValArr) == 1 || rootOpKey.Operation == request.FilterOpAnd && !skipRoot {
flat := normalizeConditions(conditionsArrToMap(rootValArr), false)
flatMap := flat.(map[connor.FilterKey]any)
for k, v := range flatMap {
addNormalizedCondition(k, v, result)
}
} else {
resultArr := []any{}
for i := range rootValArr {
norm := normalizeConditions(rootValArr[i], !skipRoot)
normMap, ok := norm.(map[connor.FilterKey]any)
if ok {
for k, v := range normMap {
resultArr = append(resultArr, map[connor.FilterKey]any{k: v})
}
} else {
resultArr = append(resultArr, norm)
}
}
addNormalizedCondition(rootKey, resultArr, result)
}
} else if rootOpKey.Operation == request.FilterOpNot {
notMap := rootVal.(map[connor.FilterKey]any)
if len(notMap) == 1 {
var k connor.FilterKey
for k = range notMap {
break
}
norm := normalizeConditions(notMap, true).(map[connor.FilterKey]any)
delete(notMap, k)
var v any
for k, v = range norm {
break
}
if opKey, ok := k.(*mapper.Operator); ok && opKey.Operation == request.FilterOpNot {
notNotMap := normalizeConditions(v, false).(map[connor.FilterKey]any)
for notNotKey, notNotVal := range notNotMap {
addNormalizedCondition(notNotKey, notNotVal, result)
}
} else {
notMap[k] = v
addNormalizedCondition(rootOpKey, notMap, result)
}
} else {
addNormalizedCondition(rootKey, rootVal, result)
}
} else {
addNormalizedCondition(rootKey, rootVal, result)
}
// normalizeProperty flattens and groups property filters where possible.
//
// Filters targeting the same property will be grouped into a single _and.
func normalizeProperties(parentKey connor.FilterKey, conditions []any) []any {
var merge []any
var result []any

// can only merge _and groups if parent is not an _or operator
parentOp, isParentOp := parentKey.(*mapper.Operator)
canMergeAnd := !isParentOp || parentOp.Operation != request.FilterOpOr

// accumulate properties that can be merged into a single _and
// if canMergeAnd is true, all _and groups will be merged
props := make(map[int][]any)
for _, c := range conditions {
for key, val := range c.(map[connor.FilterKey]any) {
op, ok := key.(*mapper.Operator)
if canMergeAnd && ok && op.Operation == request.FilterOpAnd {
merge = append(merge, val.([]any)...)
} else if prop, ok := key.(*mapper.PropertyIndex); ok {
props[prop.Index] = append(props[prop.Index], map[connor.FilterKey]any{key: val})
} else {
addNormalizedCondition(rootKey, normalizeConditions(rootVal, false), result)
result = append(result, map[connor.FilterKey]any{key: val})
}
}
}

// merge filters with duplicate keys into a single _and
for _, val := range props {
if len(val) == 1 {
// only 1 property so no merge required
result = append(result, val...)
} else {
// multiple properties require merge with _and
merge = append(merge, val...)
}
}

// nothing to merge
if len(merge) == 0 {
return result
case []any:
return conditionsArrToMap(typedConditions)
default:
return conditions
}

// merge into a single _and operator
key := &mapper.Operator{Operation: request.FilterOpAnd}
result = append(result, map[connor.FilterKey]any{key: merge})
return result
}

0 comments on commit 0ca46a0

Please sign in to comment.