2018-03-04 10:09:32 -05:00
|
|
|
require_relative '../../spec_helper'
|
|
|
|
require_relative 'fixtures/classes'
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
describe "Array#hash" do
|
|
|
|
it "returns the same fixnum for arrays with the same content" do
|
|
|
|
[].respond_to?(:hash).should == true
|
|
|
|
|
|
|
|
[[], [1, 2, 3]].each do |ary|
|
|
|
|
ary.hash.should == ary.dup.hash
|
|
|
|
ary.hash.should be_an_instance_of(Fixnum)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "properly handles recursive arrays" do
|
|
|
|
empty = ArraySpecs.empty_recursive_array
|
2019-07-27 06:40:09 -04:00
|
|
|
-> { empty.hash }.should_not raise_error
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
array = ArraySpecs.recursive_array
|
2019-07-27 06:40:09 -04:00
|
|
|
-> { array.hash }.should_not raise_error
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "returns the same hash for equal recursive arrays" do
|
|
|
|
rec = []; rec << rec
|
|
|
|
rec.hash.should == [rec].hash
|
|
|
|
rec.hash.should == [[rec]].hash
|
|
|
|
# This is because rec.eql?([[rec]])
|
|
|
|
# Remember that if two objects are eql?
|
|
|
|
# then the need to have the same hash
|
|
|
|
# Check the Array#eql? specs!
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns the same hash for equal recursive arrays through hashes" do
|
|
|
|
h = {} ; rec = [h] ; h[:x] = rec
|
|
|
|
rec.hash.should == [h].hash
|
|
|
|
rec.hash.should == [{x: rec}].hash
|
|
|
|
# Like above, this is because rec.eql?([{x: rec}])
|
|
|
|
end
|
|
|
|
|
|
|
|
it "calls to_int on result of calling hash on each element" do
|
|
|
|
ary = Array.new(5) do
|
|
|
|
obj = mock('0')
|
|
|
|
obj.should_receive(:hash).and_return(obj)
|
|
|
|
obj.should_receive(:to_int).and_return(0)
|
|
|
|
obj
|
|
|
|
end
|
|
|
|
|
|
|
|
ary.hash
|
|
|
|
|
|
|
|
|
|
|
|
hash = mock('1')
|
|
|
|
hash.should_receive(:to_int).and_return(1.hash)
|
|
|
|
|
|
|
|
obj = mock('@hash')
|
|
|
|
obj.instance_variable_set(:@hash, hash)
|
|
|
|
def obj.hash() @hash end
|
|
|
|
|
|
|
|
[obj].hash.should == [1].hash
|
|
|
|
end
|
|
|
|
|
|
|
|
it "ignores array class differences" do
|
|
|
|
ArraySpecs::MyArray[].hash.should == [].hash
|
|
|
|
ArraySpecs::MyArray[1, 2].hash.should == [1, 2].hash
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns same hash code for arrays with the same content" do
|
|
|
|
a = [1, 2, 3, 4]
|
|
|
|
a.fill 'a', 0..3
|
|
|
|
b = %w|a a a a|
|
|
|
|
a.hash.should == b.hash
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns the same value if arrays are #eql?" do
|
|
|
|
a = [1, 2, 3, 4]
|
|
|
|
a.fill 'a', 0..3
|
|
|
|
b = %w|a a a a|
|
|
|
|
a.hash.should == b.hash
|
|
|
|
a.should eql(b)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "produces different hashes for nested arrays with different values and empty terminator" do
|
|
|
|
[1, [1, []]].hash.should_not == [2, [2, []]].hash
|
|
|
|
end
|
|
|
|
end
|