summaryrefslogtreecommitdiff
path: root/spec/lib/gitlab/pagination/keyset/iterator_spec.rb
blob: 656ae73945ee4775f4c10cf970b10825efc7775e (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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Gitlab::Pagination::Keyset::Iterator do
  let_it_be(:project) { create(:project) }
  let_it_be(:issue_list_with_same_pos) { create_list(:issue, 3, project: project, relative_position: 100, updated_at: 1.day.ago) }
  let_it_be(:issue_list_with_null_pos) { create_list(:issue, 3, project: project, relative_position: nil, updated_at: 1.day.ago) }
  let_it_be(:issue_list_with_asc_pos) { create_list(:issue, 3, :with_asc_relative_position, project: project, updated_at: 1.day.ago) }

  let(:klass) { Issue }
  let(:column) { 'relative_position' }
  let(:direction) { :asc }
  let(:reverse_direction) { ::Gitlab::Pagination::Keyset::ColumnOrderDefinition::REVERSED_ORDER_DIRECTIONS[direction] }
  let(:nulls_position) { :nulls_last }
  let(:reverse_nulls_position) { ::Gitlab::Pagination::Keyset::ColumnOrderDefinition::REVERSED_NULL_POSITIONS[nulls_position] }
  let(:custom_reorder) do
    Gitlab::Pagination::Keyset::Order.build([
      Gitlab::Pagination::Keyset::ColumnOrderDefinition.new(
        attribute_name: column,
         column_expression: klass.arel_table[column],
         order_expression: ::Gitlab::Database.nulls_order(column, direction, nulls_position),
         reversed_order_expression: ::Gitlab::Database.nulls_order(column, reverse_direction, reverse_nulls_position),
         order_direction: direction,
         nullable: nulls_position,
         distinct: false
      ),
      Gitlab::Pagination::Keyset::ColumnOrderDefinition.new(
        attribute_name: 'id',
         order_expression: klass.arel_table[:id].send(direction),
         add_to_projections: true
      )
    ])
  end

  let(:scope) { project.issues.reorder(custom_reorder) }

  subject { described_class.new(scope: scope) }

  describe '.each_batch' do
    it 'yields an ActiveRecord::Relation when a block is given' do
      subject.each_batch(of: 1) do |relation|
        expect(relation).to be_a_kind_of(ActiveRecord::Relation)
      end
    end

    it 'accepts a custom batch size' do
      count = 0

      subject.each_batch(of: 2) { |relation| count += relation.count(:all) }

      expect(count).to eq(9)
    end

    it 'allows updating of the yielded relations' do
      time = Time.current

      subject.each_batch(of: 2) do |relation|
        relation.update_all(updated_at: time)
      end

      expect(Issue.where(updated_at: time).count).to eq(9)
    end

    context 'with ordering direction' do
      context 'when ordering asc' do
        it 'orders ascending by default, including secondary order column' do
          positions = []

          subject.each_batch(of: 2) { |rel| positions.concat(rel.pluck(:relative_position, :id)) }

          expect(positions).to eq(project.issues.order_relative_position_asc.order(id: :asc).pluck(:relative_position, :id))
        end
      end

      context 'when reversing asc order' do
        let(:scope) { project.issues.order(custom_reorder.reversed_order) }

        it 'orders in reverse of ascending' do
          positions = []

          subject.each_batch(of: 2) { |rel| positions.concat(rel.pluck(:relative_position, :id)) }

          expect(positions).to eq(project.issues.order_relative_position_desc.order(id: :desc).pluck(:relative_position, :id))
        end
      end

      context 'when asc order, with nulls first' do
        let(:nulls_position) { :nulls_first }

        it 'orders ascending with nulls first' do
          positions = []

          subject.each_batch(of: 2) { |rel| positions.concat(rel.pluck(:relative_position, :id)) }

          expect(positions).to eq(project.issues.reorder(::Gitlab::Database.nulls_first_order('relative_position', 'ASC')).order(id: :asc).pluck(:relative_position, :id))
        end
      end

      context 'when ordering desc' do
        let(:direction) { :desc }
        let(:nulls_position) { :nulls_last }

        it 'orders descending' do
          positions = []

          subject.each_batch(of: 2) { |rel| positions.concat(rel.pluck(:relative_position, :id)) }

          expect(positions).to eq(project.issues.reorder(::Gitlab::Database.nulls_last_order('relative_position', 'DESC')).order(id: :desc).pluck(:relative_position, :id))
        end
      end

      context 'when ordering by columns are repeated twice' do
        let(:direction) { :desc }
        let(:column) { :id }

        it 'orders descending' do
          positions = []

          subject.each_batch(of: 2) { |rel| positions.concat(rel.pluck(:id)) }

          expect(positions).to eq(project.issues.reorder(id: :desc).pluck(:id))
        end
      end
    end
  end
end