be_element_of_matcher.go 1.3 KB
// untested sections: 1

package matchers

import (
	"fmt"
	"reflect"

	"github.com/onsi/gomega/format"
)

type BeElementOfMatcher struct {
	Elements []interface{}
}

func (matcher *BeElementOfMatcher) Match(actual interface{}) (success bool, err error) {
	if reflect.TypeOf(actual) == nil {
		return false, fmt.Errorf("BeElement matcher expects actual to be typed")
	}

	length := len(matcher.Elements)
	valueAt := func(i int) interface{} {
		return matcher.Elements[i]
	}
	// Special handling of a single element of type Array or Slice
	if length == 1 && isArrayOrSlice(valueAt(0)) {
		element := valueAt(0)
		value := reflect.ValueOf(element)
		length = value.Len()
		valueAt = func(i int) interface{} {
			return value.Index(i).Interface()
		}
	}

	var lastError error
	for i := 0; i < length; i++ {
		matcher := &EqualMatcher{Expected: valueAt(i)}
		success, err := matcher.Match(actual)
		if err != nil {
			lastError = err
			continue
		}
		if success {
			return true, nil
		}
	}

	return false, lastError
}

func (matcher *BeElementOfMatcher) FailureMessage(actual interface{}) (message string) {
	return format.Message(actual, "to be an element of", matcher.Elements)
}

func (matcher *BeElementOfMatcher) NegatedFailureMessage(actual interface{}) (message string) {
	return format.Message(actual, "not to be an element of", matcher.Elements)
}