mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-18 10:49:30 +00:00
86a7acc2fb
Bumps [github.com/onsi/gomega](https://github.com/onsi/gomega) from 1.27.4 to 1.27.6. - [Release notes](https://github.com/onsi/gomega/releases) - [Changelog](https://github.com/onsi/gomega/blob/master/CHANGELOG.md) - [Commits](https://github.com/onsi/gomega/compare/v1.27.4...v1.27.6) --- updated-dependencies: - dependency-name: github.com/onsi/gomega dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com>
154 lines
4.0 KiB
Go
154 lines
4.0 KiB
Go
// untested sections: 3
|
|
|
|
package matchers
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
|
|
"github.com/onsi/gomega/format"
|
|
"github.com/onsi/gomega/matchers/support/goraph/bipartitegraph"
|
|
)
|
|
|
|
type ConsistOfMatcher struct {
|
|
Elements []interface{}
|
|
missingElements []interface{}
|
|
extraElements []interface{}
|
|
}
|
|
|
|
func (matcher *ConsistOfMatcher) Match(actual interface{}) (success bool, err error) {
|
|
if !isArrayOrSlice(actual) && !isMap(actual) {
|
|
return false, fmt.Errorf("ConsistOf matcher expects an array/slice/map. Got:\n%s", format.Object(actual, 1))
|
|
}
|
|
|
|
matchers := matchers(matcher.Elements)
|
|
values := valuesOf(actual)
|
|
|
|
bipartiteGraph, err := bipartitegraph.NewBipartiteGraph(values, matchers, neighbours)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
edges := bipartiteGraph.LargestMatching()
|
|
if len(edges) == len(values) && len(edges) == len(matchers) {
|
|
return true, nil
|
|
}
|
|
|
|
var missingMatchers []interface{}
|
|
matcher.extraElements, missingMatchers = bipartiteGraph.FreeLeftRight(edges)
|
|
matcher.missingElements = equalMatchersToElements(missingMatchers)
|
|
|
|
return false, nil
|
|
}
|
|
|
|
func neighbours(value, matcher interface{}) (bool, error) {
|
|
match, err := matcher.(omegaMatcher).Match(value)
|
|
return match && err == nil, nil
|
|
}
|
|
|
|
func equalMatchersToElements(matchers []interface{}) (elements []interface{}) {
|
|
for _, matcher := range matchers {
|
|
if equalMatcher, ok := matcher.(*EqualMatcher); ok {
|
|
elements = append(elements, equalMatcher.Expected)
|
|
} else if _, ok := matcher.(*BeNilMatcher); ok {
|
|
elements = append(elements, nil)
|
|
} else {
|
|
elements = append(elements, matcher)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func flatten(elems []interface{}) []interface{} {
|
|
if len(elems) != 1 || !isArrayOrSlice(elems[0]) {
|
|
return elems
|
|
}
|
|
|
|
value := reflect.ValueOf(elems[0])
|
|
flattened := make([]interface{}, value.Len())
|
|
for i := 0; i < value.Len(); i++ {
|
|
flattened[i] = value.Index(i).Interface()
|
|
}
|
|
return flattened
|
|
}
|
|
|
|
func matchers(expectedElems []interface{}) (matchers []interface{}) {
|
|
for _, e := range flatten(expectedElems) {
|
|
if e == nil {
|
|
matchers = append(matchers, &BeNilMatcher{})
|
|
} else if matcher, isMatcher := e.(omegaMatcher); isMatcher {
|
|
matchers = append(matchers, matcher)
|
|
} else {
|
|
matchers = append(matchers, &EqualMatcher{Expected: e})
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func presentable(elems []interface{}) interface{} {
|
|
elems = flatten(elems)
|
|
|
|
if len(elems) == 0 {
|
|
return []interface{}{}
|
|
}
|
|
|
|
sv := reflect.ValueOf(elems)
|
|
firstEl := sv.Index(0)
|
|
if firstEl.IsNil() {
|
|
return elems
|
|
}
|
|
tt := firstEl.Elem().Type()
|
|
for i := 1; i < sv.Len(); i++ {
|
|
el := sv.Index(i)
|
|
if el.IsNil() || (sv.Index(i).Elem().Type() != tt) {
|
|
return elems
|
|
}
|
|
}
|
|
|
|
ss := reflect.MakeSlice(reflect.SliceOf(tt), sv.Len(), sv.Len())
|
|
for i := 0; i < sv.Len(); i++ {
|
|
ss.Index(i).Set(sv.Index(i).Elem())
|
|
}
|
|
|
|
return ss.Interface()
|
|
}
|
|
|
|
func valuesOf(actual interface{}) []interface{} {
|
|
value := reflect.ValueOf(actual)
|
|
values := []interface{}{}
|
|
if isMap(actual) {
|
|
keys := value.MapKeys()
|
|
for i := 0; i < value.Len(); i++ {
|
|
values = append(values, value.MapIndex(keys[i]).Interface())
|
|
}
|
|
} else {
|
|
for i := 0; i < value.Len(); i++ {
|
|
values = append(values, value.Index(i).Interface())
|
|
}
|
|
}
|
|
|
|
return values
|
|
}
|
|
|
|
func (matcher *ConsistOfMatcher) FailureMessage(actual interface{}) (message string) {
|
|
message = format.Message(actual, "to consist of", presentable(matcher.Elements))
|
|
message = appendMissingElements(message, matcher.missingElements)
|
|
if len(matcher.extraElements) > 0 {
|
|
message = fmt.Sprintf("%s\nthe extra elements were\n%s", message,
|
|
format.Object(presentable(matcher.extraElements), 1))
|
|
}
|
|
return
|
|
}
|
|
|
|
func appendMissingElements(message string, missingElements []interface{}) string {
|
|
if len(missingElements) == 0 {
|
|
return message
|
|
}
|
|
return fmt.Sprintf("%s\nthe missing elements were\n%s", message,
|
|
format.Object(presentable(missingElements), 1))
|
|
}
|
|
|
|
func (matcher *ConsistOfMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
|
return format.Message(actual, "not to consist of", presentable(matcher.Elements))
|
|
}
|