58 lines
2.4 KiB
Ruby
58 lines
2.4 KiB
Ruby
|
# frozen_string_literal: true
|
||
|
|
||
|
module Gitlab
|
||
|
module Graphql
|
||
|
module Pagination
|
||
|
module Keyset
|
||
|
# This class handles the last(N) ActiveRecord call even if a special ORDER BY configuration is present.
|
||
|
# For the last(N) call, ActiveRecord calls reverse_order, however for some cases it raises
|
||
|
# ActiveRecord::IrreversibleOrderError error.
|
||
|
class LastItems
|
||
|
# rubocop: disable CodeReuse/ActiveRecord
|
||
|
def self.take_items(scope, count)
|
||
|
if custom_order = lookup_custom_reverse_order(scope.order_values)
|
||
|
items = scope.reorder(*custom_order).first(count) # returns a single record when count is nil
|
||
|
items.is_a?(Array) ? items.reverse : items
|
||
|
else
|
||
|
scope.last(count)
|
||
|
end
|
||
|
end
|
||
|
# rubocop: enable CodeReuse/ActiveRecord
|
||
|
|
||
|
# Detect special ordering and provide the reversed order
|
||
|
def self.lookup_custom_reverse_order(order_values)
|
||
|
if ordering_by_merged_at_and_mr_id_desc?(order_values)
|
||
|
[
|
||
|
Gitlab::Database.nulls_first_order('merge_request_metrics.merged_at', 'ASC'), # reversing the order
|
||
|
MergeRequest.arel_table[:id].asc
|
||
|
]
|
||
|
elsif ordering_by_merged_at_and_mr_id_asc?(order_values)
|
||
|
[
|
||
|
Gitlab::Database.nulls_first_order('merge_request_metrics.merged_at', 'DESC'),
|
||
|
MergeRequest.arel_table[:id].asc
|
||
|
]
|
||
|
end
|
||
|
end
|
||
|
|
||
|
def self.ordering_by_merged_at_and_mr_id_desc?(order_values)
|
||
|
order_values.size == 2 &&
|
||
|
order_values.first.to_s == Gitlab::Database.nulls_last_order('merge_request_metrics.merged_at', 'DESC') &&
|
||
|
order_values.last.is_a?(Arel::Nodes::Descending) &&
|
||
|
order_values.last.to_sql == MergeRequest.arel_table[:id].desc.to_sql
|
||
|
end
|
||
|
|
||
|
def self.ordering_by_merged_at_and_mr_id_asc?(order_values)
|
||
|
order_values.size == 2 &&
|
||
|
order_values.first.to_s == Gitlab::Database.nulls_last_order('merge_request_metrics.merged_at', 'ASC') &&
|
||
|
order_values.last.is_a?(Arel::Nodes::Descending) &&
|
||
|
order_values.last.to_sql == MergeRequest.arel_table[:id].desc.to_sql
|
||
|
end
|
||
|
|
||
|
private_class_method :ordering_by_merged_at_and_mr_id_desc?
|
||
|
private_class_method :ordering_by_merged_at_and_mr_id_asc?
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|