module RSpec module Matchers class MatchArray include RSpec::Matchers::Pretty def initialize(expected) @expected = expected end def matches?(actual) @actual = actual @extra_items = difference_between_arrays(@actual, @expected) @missing_items = difference_between_arrays(@expected, @actual) @extra_items.empty? & @missing_items.empty? end def failure_message_for_should message = "expected collection contained: #{safe_sort(@expected).inspect}\n" message += "actual collection contained: #{safe_sort(@actual).inspect}\n" message += "the missing elements were: #{safe_sort(@missing_items).inspect}\n" unless @missing_items.empty? message += "the extra elements were: #{safe_sort(@extra_items).inspect}\n" unless @extra_items.empty? message end def failure_message_for_should_not "Matcher does not support should_not" end def description "contain exactly #{_pretty_print(@expected)}" end private def safe_sort(array) array.sort rescue array end def difference_between_arrays(array_1, array_2) difference = array_1.dup array_2.each do |element| if index = difference.index(element) difference.delete_at(index) end end difference end end # Passes if actual contains all of the expected regardless of order. # This works for collections. Pass in multiple args and it will only # pass if all args are found in collection. # # NOTE: there is no should_not version of array.should =~ other_array # # @example # # [1,2,3].should =~ [1,2,3] # => would pass # [1,2,3].should =~ [2,3,1] # => would pass # [1,2,3,4].should =~ [1,2,3] # => would fail # [1,2,2,3].should =~ [1,2,3] # => would fail # [1,2,3].should =~ [1,2,3,4] # => would fail OperatorMatcher.register(Array, '=~', RSpec::Matchers::MatchArray) end end