summaryrefslogtreecommitdiff
path: root/spec/ruby/core/array/sort_by_spec.rb
blob: 0334f953f6c20a1cc253470860884007fae30a08 (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
require_relative '../../spec_helper'
require_relative 'fixtures/classes'
require_relative 'shared/iterable_and_tolerating_size_increasing'
require_relative '../enumerable/shared/enumeratorized'

describe "Array#sort_by!" do
  it "sorts array in place by passing each element to the given block" do
    a = [-100, -2, 1, 200, 30000]
    a.sort_by!{ |e| e.to_s.size }
    a.should == [1, -2, 200, -100, 30000]
  end

  it "returns an Enumerator if not given a block" do
    (1..10).to_a.sort_by!.should be_an_instance_of(Enumerator)
  end

  it "completes when supplied a block that always returns the same result" do
    a = [2, 3, 5, 1, 4]
    a.sort_by!{  1 }
    a.should be_an_instance_of(Array)
    a.sort_by!{  0 }
    a.should be_an_instance_of(Array)
    a.sort_by!{ -1 }
    a.should be_an_instance_of(Array)
  end

  it "raises a FrozenError on a frozen array" do
    -> { ArraySpecs.frozen_array.sort_by! {}}.should raise_error(FrozenError)
  end

  it "raises a FrozenError on an empty frozen array" do
    -> { ArraySpecs.empty_frozen_array.sort_by! {}}.should raise_error(FrozenError)
  end

  it "raises a FrozenError on a frozen array only during iteration if called without a block" do
    enum = ArraySpecs.frozen_array.sort_by!
    -> { enum.each {} }.should raise_error(FrozenError)
  end

  it "returns the specified value when it would break in the given block" do
    [1, 2, 3].sort_by!{ break :a }.should == :a
  end

  it "makes some modification even if finished sorting when it would break in the given block" do
    partially_sorted = (1..5).map{|i|
      ary = [5, 4, 3, 2, 1]
      ary.sort_by!{|x,y| break if x==i; x<=>y}
      ary
    }
    partially_sorted.any?{|ary| ary != [1, 2, 3, 4, 5]}.should be_true
  end

  it "changes nothing when called on a single element array" do
    [1].sort_by!(&:to_s).should == [1]
  end

  it "does not truncate the array is the block raises an exception" do
    a = [1, 2, 3]
    begin
      a.sort_by! { raise StandardError, 'Oops' }
    rescue
    end

    a.should == [1, 2, 3]
  end

  it "doesn't change array if error is raised" do
    a = [4, 3, 2, 1]
    begin
      a.sort_by! do |e|
        raise StandardError, 'Oops' if e == 1
        e
      end
    rescue StandardError
    end

    a.should == [4, 3, 2, 1]
  end

  it_behaves_like :enumeratorized_with_origin_size, :sort_by!, [1,2,3]
end

describe "Array#sort_by!" do
  it_behaves_like :array_iterable_and_tolerating_size_increasing, :sort_by!
end