require 'spec_helper' describe Duck do before { 200.times do |i| Duck.create \ :name => "Duck #{i}" end } describe "setting and fetching by position" do describe '137' do before { @last = Duck.last @last.update_attribute :row_position, 137 } subject { Duck.ranker(:row).with(Duck.new).current_at_position(137).instance } its(:id) { should == @last.id } end describe '2' do before { @last = Duck.last @last.update_attribute :row_position, 2 } subject { Duck.ranker(:row).with(Duck.new).current_at_position(2).instance } its(:id) { should == @last.id } end describe 'last' do before { @last = Duck.last @last.update_attribute :row_position, :last } subject { Duck.rank(:row).last } its(:id) { should == @last.id } end describe 'first' do before { @last = Duck.last @last.update_attribute :row_position, :first } subject { Duck.rank(:row).first } its(:id) { should == @last.id } end end describe "a rearrangement" do describe "with max value" do before { @first = Duck.first @second = Duck.offset(1).first @ordered = Duck.rank(:row).where(%Q{"id" NOT IN (#{@first.id}, #{@second.id})}).collect {|d| d.id } @first.update_attribute :row, RankedModel::MAX_RANK_VALUE @second.update_attribute :row, RankedModel::MAX_RANK_VALUE } context { subject { Duck.rank(:row).collect {|d| d.id } } it { should == (@ordered[0..-2] + [@ordered[-1], @first.id, @second.id]) } } end describe "with max value and with_same pond" do before { Duck.order("id ASC").limit(50).each_with_index do |d, index| d.update_attributes :age => index % 10, :pond => "Pond #{index / 10}" end @duck_11 = Duck.offset(10).first @duck_12 = Duck.offset(11).first @ordered = Duck.where(:pond => 'Pond 1').rank(:age).where(%Q{"id" NOT IN (#{@duck_11.id}, #{@duck_12.id})}).collect {|d| d.id } @duck_11.update_attribute :age, RankedModel::MAX_RANK_VALUE @duck_12.update_attribute :age, RankedModel::MAX_RANK_VALUE } context { subject { Duck.where(:pond => 'Pond 1').rank(:age).collect {|d| d.id } } it { should == (@ordered[0..-2] + [@ordered[-1], @duck_11.id, @duck_12.id]) } } context { subject { Duck.first.age } it { should == 0} } end describe "with min value" do before { @first = Duck.first @second = Duck.offset(1).first @ordered = Duck.rank(:row).where(%Q{"id" NOT IN (#{@first.id}, #{@second.id})}).collect {|d| d.id } @first.update_attribute :row, RankedModel::MIN_RANK_VALUE @second.update_attribute :row, RankedModel::MIN_RANK_VALUE } context { subject { Duck.rank(:row).collect {|d| d.id } } it { should == ([@second.id, @first.id] + @ordered) } } end describe "with no more gaps" do before { @first = Duck.first @second = Duck.where(:row => RankedModel::MAX_RANK_VALUE).first || Duck.offset(1).first @third = Duck.offset(2).first @fourth = Duck.offset(4).first @lower = Duck.rank(:row). where(%Q{"id" NOT IN (#{@first.id}, #{@second.id}, #{@third.id}, #{@fourth.id})}). where(%Q{"row" < ?}, RankedModel::MAX_RANK_VALUE / 2). collect {|d| d.id } @upper = Duck.rank(:row). where(%Q{"id" NOT IN (#{@first.id}, #{@second.id}, #{@third.id}, #{@fourth.id})}). where(%Q{"row" >= ?}, RankedModel::MAX_RANK_VALUE / 2). collect {|d| d.id } @first.update_attribute :row, RankedModel::MIN_RANK_VALUE @second.update_attribute :row, RankedModel::MAX_RANK_VALUE @third.update_attribute :row, (RankedModel::MAX_RANK_VALUE / 2) @fourth.update_attribute :row, @third.row } context { subject { Duck.rank(:row).collect {|d| d.id } } it { should == ([@first.id] + @lower + [@fourth.id, @third.id] + @upper + [@second.id]) } } end end end