summaryrefslogtreecommitdiff
path: root/spec/ruby/core/enumerable/uniq_spec.rb
blob: eb1e70c2085f6c1ab79aa26746203a36d8ee3a12 (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
require_relative '../../spec_helper'
require_relative 'fixtures/classes'

ruby_version_is '2.4' do
  describe 'Enumerable#uniq' do
    it 'returns an array that contains only unique elements' do
      [0, 1, 2, 3].to_enum.uniq { |n| n.even? }.should == [0, 1]
    end

    it "uses eql? semantics" do
      [1.0, 1].to_enum.uniq.should == [1.0, 1]
    end

    it "compares elements first with hash" do
      x = mock('0')
      x.should_receive(:hash).at_least(1).and_return(0)
      y = mock('0')
      y.should_receive(:hash).at_least(1).and_return(0)

      [x, y].to_enum.uniq.should == [x, y]
    end

    it "does not compare elements with different hash codes via eql?" do
      x = mock('0')
      x.should_not_receive(:eql?)
      y = mock('1')
      y.should_not_receive(:eql?)

      x.should_receive(:hash).at_least(1).and_return(0)
      y.should_receive(:hash).at_least(1).and_return(1)

      [x, y].to_enum.uniq.should == [x, y]
    end

    it "compares elements with matching hash codes with #eql?" do
      a = Array.new(2) do
        obj = mock('0')
        obj.should_receive(:hash).at_least(1).and_return(0)

        def obj.eql?(o)
          # It's undefined whether the impl does a[0].eql?(a[1]) or
          # a[1].eql?(a[0]) so we taint both.
          taint
          o.taint
          false
        end

        obj
      end

      a.uniq.should == a
      a[0].tainted?.should == true
      a[1].tainted?.should == true

      a = Array.new(2) do
        obj = mock('0')
        obj.should_receive(:hash).at_least(1).and_return(0)

        def obj.eql?(o)
          # It's undefined whether the impl does a[0].eql?(a[1]) or
          # a[1].eql?(a[0]) so we taint both.
          taint
          o.taint
          true
        end

        obj
      end

      a.to_enum.uniq.size.should == 1
      a[0].tainted?.should == true
      a[1].tainted?.should == true
    end

    context 'when yielded with multiple arguments' do
      before :each do
        @enum = Object.new.to_enum
        class << @enum
          def each
            yield 0, 'foo'
            yield 1, 'FOO'
            yield 2, 'bar'
          end
        end
      end

      ruby_bug '#13669', ''...'2.5' do
        it 'returns all yield arguments as an array' do
          @enum.uniq { |_, label| label.downcase }.should == [[0, 'foo'], [2, 'bar']]
        end
      end
    end
  end
end