summaryrefslogtreecommitdiff
path: root/lib/gitlab/graphql/pagination/keyset
diff options
context:
space:
mode:
Diffstat (limited to 'lib/gitlab/graphql/pagination/keyset')
-rw-r--r--lib/gitlab/graphql/pagination/keyset/connection.rb1
-rw-r--r--lib/gitlab/graphql/pagination/keyset/generic_keyset_pagination.rb47
-rw-r--r--lib/gitlab/graphql/pagination/keyset/last_items.rb38
-rw-r--r--lib/gitlab/graphql/pagination/keyset/order_info.rb7
4 files changed, 51 insertions, 42 deletions
diff --git a/lib/gitlab/graphql/pagination/keyset/connection.rb b/lib/gitlab/graphql/pagination/keyset/connection.rb
index f95c91c5706..e525996ec10 100644
--- a/lib/gitlab/graphql/pagination/keyset/connection.rb
+++ b/lib/gitlab/graphql/pagination/keyset/connection.rb
@@ -33,6 +33,7 @@ module Gitlab
include Gitlab::Utils::StrongMemoize
include ::Gitlab::Graphql::ConnectionCollectionMethods
prepend ::Gitlab::Graphql::ConnectionRedaction
+ prepend GenericKeysetPagination
# rubocop: disable Naming/PredicateName
# https://relay.dev/graphql/connections.htm#sec-undefined.PageInfo.Fields
diff --git a/lib/gitlab/graphql/pagination/keyset/generic_keyset_pagination.rb b/lib/gitlab/graphql/pagination/keyset/generic_keyset_pagination.rb
new file mode 100644
index 00000000000..318c6e1734f
--- /dev/null
+++ b/lib/gitlab/graphql/pagination/keyset/generic_keyset_pagination.rb
@@ -0,0 +1,47 @@
+# frozen_string_literal: true
+
+module Gitlab
+ module Graphql
+ module Pagination
+ module Keyset
+ # Use the generic keyset implementation if the given ActiveRecord scope supports it.
+ # Note: this module is temporary, at some point it will be merged with Keyset::Connection
+ module GenericKeysetPagination
+ extend ActiveSupport::Concern
+
+ def ordered_items
+ return super unless Gitlab::Pagination::Keyset::Order.keyset_aware?(items)
+
+ items
+ end
+
+ def cursor_for(node)
+ return super unless Gitlab::Pagination::Keyset::Order.keyset_aware?(items)
+
+ order = Gitlab::Pagination::Keyset::Order.extract_keyset_order_object(items)
+ encode(order.cursor_attributes_for_node(node).to_json)
+ end
+
+ def slice_nodes(sliced, encoded_cursor, before_or_after)
+ return super unless Gitlab::Pagination::Keyset::Order.keyset_aware?(sliced)
+
+ order = Gitlab::Pagination::Keyset::Order.extract_keyset_order_object(sliced)
+ order = order.reversed_order if before_or_after == :before
+
+ decoded_cursor = ordering_from_encoded_json(encoded_cursor)
+ order.apply_cursor_conditions(sliced, decoded_cursor)
+ end
+
+ def sliced_nodes
+ return super unless Gitlab::Pagination::Keyset::Order.keyset_aware?(items)
+
+ sliced = ordered_items
+ sliced = slice_nodes(sliced, before, :before) if before.present?
+ sliced = slice_nodes(sliced, after, :after) if after.present?
+ sliced
+ end
+ end
+ end
+ end
+ end
+end
diff --git a/lib/gitlab/graphql/pagination/keyset/last_items.rb b/lib/gitlab/graphql/pagination/keyset/last_items.rb
index 45bf15236c1..960567a6fbc 100644
--- a/lib/gitlab/graphql/pagination/keyset/last_items.rb
+++ b/lib/gitlab/graphql/pagination/keyset/last_items.rb
@@ -10,46 +10,14 @@ module Gitlab
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
+ if Gitlab::Pagination::Keyset::Order.keyset_aware?(scope)
+ order = Gitlab::Pagination::Keyset::Order.extract_keyset_order_object(scope)
+ items = scope.reorder(order.reversed_order).first(count)
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
diff --git a/lib/gitlab/graphql/pagination/keyset/order_info.rb b/lib/gitlab/graphql/pagination/keyset/order_info.rb
index d37264c1343..0494329bfd9 100644
--- a/lib/gitlab/graphql/pagination/keyset/order_info.rb
+++ b/lib/gitlab/graphql/pagination/keyset/order_info.rb
@@ -92,8 +92,6 @@ module Gitlab
def extract_attribute_values(order_value)
if ordering_by_lower?(order_value)
[order_value.expr.expressions[0].name.to_s, order_value.direction, order_value.expr]
- elsif ordering_by_similarity?(order_value)
- ['similarity', order_value.direction, order_value.expr]
elsif ordering_by_case?(order_value)
['case_order_value', order_value.direction, order_value.expr]
elsif ordering_by_array_position?(order_value)
@@ -113,11 +111,6 @@ module Gitlab
order_value.expr.is_a?(Arel::Nodes::NamedFunction) && order_value.expr&.name&.downcase == 'array_position'
end
- # determine if ordering using SIMILARITY scoring based on Gitlab::Database::SimilarityScore
- def ordering_by_similarity?(order_value)
- Gitlab::Database::SimilarityScore.order_by_similarity?(order_value)
- end
-
# determine if ordering using CASE
def ordering_by_case?(order_value)
order_value.expr.is_a?(Arel::Nodes::Case)