summaryrefslogtreecommitdiff
path: root/spec/ruby/core/hash/dig_spec.rb
blob: 91639f42130af03a0736f03b4325c4252d0c2c9a (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
require_relative '../../spec_helper'

ruby_version_is '2.3' do
  describe "Hash#dig" do

    it "returns #[] with one arg" do
      h = { 0 => false, a: 1 }
      h.dig(:a).should == 1
      h.dig(0).should be_false
      h.dig(1).should be_nil
    end

    it "returns the nested value specified by the sequence of keys" do
      h = { foo: { bar: { baz: 1 } } }
      h.dig(:foo, :bar, :baz).should == 1
      h.dig(:foo, :bar, :nope).should be_nil
      h.dig(:foo, :baz).should be_nil
      h.dig(:bar, :baz, :foo).should be_nil
    end

    it "returns the nested value specified if the sequence includes an index" do
      h = { foo: [1, 2, 3] }
      h.dig(:foo, 2).should == 3
    end

    it "returns nil if any intermediate step is nil" do
      h = { foo: { bar: { baz: 1 } } }
      h.dig(:foo, :zot, :xyz).should == nil
    end

    it "raises an ArgumentError if no arguments provided" do
      lambda { { the: 'borg' }.dig() }.should raise_error(ArgumentError)
    end

    it "handles type-mixed deep digging" do
      h = {}
      h[:foo] = [ { bar: [ 1 ] }, [ obj = Object.new, 'str' ] ]
      def obj.dig(*args); [ 42 ] end

      h.dig(:foo, 0, :bar).should == [ 1 ]
      h.dig(:foo, 0, :bar, 0).should == 1
      h.dig(:foo, 1, 1).should == 'str'
      # MRI does not recurse values returned from `obj.dig`
      h.dig(:foo, 1, 0, 0).should == [ 42 ]
      h.dig(:foo, 1, 0, 0, 10).should == [ 42 ]
    end

    it "raises TypeError if an intermediate element does not respond to #dig" do
      h = {}
      h[:foo] = [ { bar: [ 1 ] }, [ nil, 'str' ] ]
      lambda { h.dig(:foo, 0, :bar, 0, 0) }.should raise_error(TypeError)
      lambda { h.dig(:foo, 1, 1, 0) }.should raise_error(TypeError)
    end

    it "calls #dig on the result of #[] with the remaining arguments" do
      h = { foo: { bar: { baz: 42 } } }
      h[:foo].should_receive(:dig).with(:bar, :baz).and_return(42)
      h.dig(:foo, :bar, :baz).should == 42
    end

    it "respects Hash's default" do
      default = {bar: 42}
      h = Hash.new(default)
      h.dig(:foo).should equal default
      h.dig(:foo, :bar).should == 42
    end
  end
end