File: merge_spec.rb

package info (click to toggle)
ruby-immutable-ruby 0.1.0-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 1,852 kB
  • sloc: ruby: 16,556; makefile: 4
file content (82 lines) | stat: -rw-r--r-- 2,484 bytes parent folder | download
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
require 'spec_helper'

describe Immutable::Hash do
  describe '#merge' do
    [
      [{}, {}, {}],
      [{'A' => 'aye'}, {}, {'A' => 'aye'}],
      [{'A' => 'aye'}, {'A' => 'bee'}, {'A' => 'bee'}],
      [{'A' => 'aye'}, {'B' => 'bee'}, {'A' => 'aye', 'B' => 'bee'}],
      [(1..300).zip(1..300), (150..450).zip(150..450), (1..450).zip(1..450)]
    ].each do |a, b, expected|
      context "for #{a.inspect} and #{b.inspect}" do
        let(:hash_a) { H[a] }
        let(:hash_b) { H[b] }
        let(:result) { hash_a.merge(hash_b) }

        it "returns #{expected.inspect} when passed an Immutable::Hash"  do
          result.should eql(H[expected])
        end

        it "returns #{expected.inspect} when passed a Ruby Hash" do
          H[a].merge(::Hash[b]).should eql(H[expected])
        end

        it "doesn't change the original Hashes" do
          result
          hash_a.should eql(H[a])
          hash_b.should eql(H[b])
        end
      end
    end

    context 'when merging with an empty Hash' do
      it 'returns self' do
        hash = H[a: 1, b: 2]
        hash.merge(H.empty).should be(hash)
      end
    end

    context 'when merging with subset Hash' do
      it 'returns self' do
        big_hash   = H[(1..300).zip(1..300)]
        small_hash = H[(1..200).zip(1..200)]
        big_hash.merge(small_hash).should be(big_hash)
      end
    end

    context 'when called on a subclass' do
      it 'returns an instance of the subclass' do
        subclass = Class.new(Immutable::Hash)
        instance = subclass.new(a: 1, b: 2)
        instance.merge(c: 3, d: 4).class.should be(subclass)
      end
    end

    it 'sets any duplicate key to the value of block if passed a block' do
      h1 = H[a: 2, b: 1, d: 5]
      h2 = H[a: -2, b: 4, c: -3]
      r = h1.merge(h2) { |k,x,y| nil }
      r.should eql(H[a: nil, b: nil, c: -3, d: 5])

      r = h1.merge(h2) { |k,x,y| "#{k}:#{x+2*y}" }
      r.should eql(H[a: 'a:-2', b: 'b:9', c: -3, d: 5])

      lambda {
        h1.merge(h2) { |k, x, y| raise(IndexError) }
      }.should raise_error(IndexError)

      r = h1.merge(h1) { |k,x,y| :x }
      r.should eql(H[a: :x, b: :x, d: :x])
    end

    it 'yields key/value pairs in the same order as #each' do
      hash = H[a: 1, b: 2, c: 3]
      each_pairs = []
      merge_pairs = []
      hash.each { |k, v| each_pairs << [k, v] }
      hash.merge(hash) { |k, v1, v2| merge_pairs << [k, v1] }
      each_pairs.should == merge_pairs
    end
  end
end