summaryrefslogtreecommitdiff
path: root/spec/ruby/core/hash/shift_spec.rb
blob: ea36488a04dfaf017953301846795f7bd91ef124 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
require_relative '../../spec_helper'
require_relative 'fixtures/classes'

describe "Hash#shift" do
  it "removes a pair from hash and return it" do
    h = { a: 1, b: 2, "c" => 3, nil => 4, [] => 5 }
    h2 = h.dup

    h.size.times do |i|
      r = h.shift
      r.should be_kind_of(Array)
      h2[r.first].should == r.last
      h.size.should == h2.size - i - 1
    end

    h.should == {}
  end

  # MRI explicitly implements this behavior
  it "allows shifting entries while iterating" do
    h = { a: 1, b: 2, c: 3 }
    visited = []
    shifted = []
    h.each_pair { |k,v|
      visited << k
      shifted << h.shift
    }
    visited.should == [:a, :b, :c]
    shifted.should == [[:a, 1], [:b, 2], [:c, 3]]
    h.should == {}
  end

  ruby_version_is '3.2' do
    it "returns nil if the Hash is empty" do
      h = {}
      def h.default(key)
        raise
      end
      h.shift.should == nil
    end
  end

  ruby_version_is ''...'3.2' do
    it "calls #default with nil if the Hash is empty" do
      h = {}
      def h.default(key)
        key.should == nil
        :foo
      end
      h.shift.should == :foo
    end
  end

  it "returns nil from an empty hash" do
    {}.shift.should == nil
  end

  ruby_version_is '3.2' do
    it "returns nil for empty hashes with defaults and default procs" do
      Hash.new(5).shift.should == nil
      h = Hash.new { |*args| args }
      h.shift.should == nil
    end
  end

  ruby_version_is ''...'3.2' do
    it "returns (computed) default for empty hashes" do
      Hash.new(5).shift.should == 5
      h = Hash.new { |*args| args }
      h.shift.should == [h, nil]
    end
  end

  it "preserves Hash invariants when removing the last item" do
    h = { :a => 1, :b => 2 }
    h.shift.should == [:a, 1]
    h.shift.should == [:b, 2]
    h[:c] = 3
    h.should == {:c => 3}
  end

  it "raises a FrozenError if called on a frozen instance" do
    -> { HashSpecs.frozen_hash.shift  }.should raise_error(FrozenError)
    -> { HashSpecs.empty_frozen_hash.shift }.should raise_error(FrozenError)
  end

  it "works when the hash is at capacity" do
    # We try a wide range of sizes in hopes that this will cover all implementations' base Hash size.
    results = []
    1.upto(100) do |n|
      h = {}
      n.times do |i|
        h[i] = i
      end
      h.shift
      results << h.size
    end

    results.should == 0.upto(99).to_a
  end
end