debian-mirror-gitlab/lib/api/helpers/pagination.rb

257 lines
8 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2017-08-17 22:00:37 +05:30
module API
module Helpers
module Pagination
def paginate(relation)
2018-11-08 19:23:39 +05:30
strategy = if params[:pagination] == 'keyset' && Feature.enabled?('api_keyset_pagination')
KeysetPaginationStrategy
else
DefaultPaginationStrategy
end
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
strategy.new(self).paginate(relation)
2017-08-17 22:00:37 +05:30
end
2019-07-07 11:18:12 +05:30
class Base
private
def per_page
@per_page ||= params[:per_page]
end
def base_request_uri
@base_request_uri ||= URI.parse(request.url).tap do |uri|
uri.host = Gitlab.config.gitlab.host
uri.port = nil
end
end
def build_page_url(query_params:)
base_request_uri.tap do |uri|
uri.query = query_params
end.to_s
end
def page_href(next_page_params = {})
query_params = params.merge(**next_page_params, per_page: per_page).to_query
build_page_url(query_params: query_params)
end
end
2018-11-08 19:23:39 +05:30
class KeysetPaginationInfo
attr_reader :relation, :request_context
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
def initialize(relation, request_context)
# This is because it's rather complex to support multiple values with possibly different sort directions
# (and we don't need this in the API)
if relation.order_values.size > 1
raise "Pagination only supports ordering by a single column." \
"The following columns were given: #{relation.order_values.map { |v| v.expr.name }}"
end
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
@relation = relation
@request_context = request_context
end
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
def fields
keys.zip(values).reject { |_, v| v.nil? }.to_h
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
def column_for_order_by(relation)
relation.order_values.first&.expr&.name
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
# Sort direction (`:asc` or `:desc`)
def sort
@sort ||= if order_by_primary_key?
# Default order is by id DESC
:desc
else
# API defaults to DESC order if param `sort` not present
request_context.params[:sort]&.to_sym || :desc
end
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
# Do we only sort by primary key?
def order_by_primary_key?
keys.size == 1 && keys.first == primary_key
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
def primary_key
relation.model.primary_key.to_sym
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
def sort_ascending?
sort == :asc
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
# Build hash of request parameters for a given record (relevant to pagination)
def params_for(record)
return {} unless record
keys.each_with_object({}) do |key, h|
h["ks_prev_#{key}".to_sym] = record.attributes[key.to_s]
end
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
private
# All values present in request parameters that correspond to #keys.
def values
@values ||= keys.map do |key|
request_context.params["ks_prev_#{key}".to_sym]
end
end
2017-09-10 17:25:29 +05:30
2018-11-08 19:23:39 +05:30
# All keys relevant to pagination.
# This always includes the primary key. Optionally, the `order_by` key is prepended.
def keys
@keys ||= [column_for_order_by(relation), primary_key].compact.uniq
end
2017-09-10 17:25:29 +05:30
end
2018-03-17 18:26:18 +05:30
2019-07-07 11:18:12 +05:30
class KeysetPaginationStrategy < Base
2018-11-08 19:23:39 +05:30
attr_reader :request_context
delegate :params, :header, :request, to: :request_context
def initialize(request_context)
@request_context = request_context
end
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2018-11-08 19:23:39 +05:30
def paginate(relation)
pagination = KeysetPaginationInfo.new(relation, request_context)
paged_relation = relation.limit(per_page)
if conds = conditions(pagination)
paged_relation = paged_relation.where(*conds)
end
# In all cases: sort by primary key (possibly in addition to another sort column)
paged_relation = paged_relation.order(pagination.primary_key => pagination.sort)
add_default_pagination_headers
if last_record = paged_relation.last
next_page_params = pagination.params_for(last_record)
add_navigation_links(next_page_params)
end
paged_relation
end
2018-12-05 23:21:45 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2018-11-08 19:23:39 +05:30
private
def conditions(pagination)
fields = pagination.fields
2019-07-07 11:18:12 +05:30
return if fields.empty?
2018-11-08 19:23:39 +05:30
placeholder = fields.map { '?' }
comp = if pagination.sort_ascending?
'>'
else
'<'
end
[
# Row value comparison:
# (A, B) < (a, b) <=> (A < a) OR (A = a AND B < b)
# <=> A <= a AND ((A < a) OR (A = a AND B < b))
"(#{fields.keys.join(',')}) #{comp} (#{placeholder.join(',')})",
*fields.values
]
end
def add_default_pagination_headers
2019-03-02 22:35:43 +05:30
header 'X-Per-Page', per_page.to_s
2018-11-08 19:23:39 +05:30
end
def add_navigation_links(next_page_params)
header 'X-Next-Page', page_href(next_page_params)
header 'Link', link_for('next', next_page_params)
2018-03-17 18:26:18 +05:30
end
2018-11-08 19:23:39 +05:30
def link_for(rel, next_page_params)
%(<#{page_href(next_page_params)}>; rel="#{rel}")
end
2018-03-17 18:26:18 +05:30
end
2019-07-07 11:18:12 +05:30
class DefaultPaginationStrategy < Base
2018-11-08 19:23:39 +05:30
attr_reader :request_context
delegate :params, :header, :request, to: :request_context
def initialize(request_context)
@request_context = request_context
end
def paginate(relation)
2019-03-02 22:35:43 +05:30
paginate_with_limit_optimization(add_default_order(relation)).tap do |data|
2018-11-08 19:23:39 +05:30
add_pagination_headers(data)
end
end
private
2019-03-02 22:35:43 +05:30
def paginate_with_limit_optimization(relation)
pagination_data = relation.page(params[:page]).per(params[:per_page])
return pagination_data unless pagination_data.is_a?(ActiveRecord::Relation)
return pagination_data unless Feature.enabled?(:api_kaminari_count_with_limit)
limited_total_count = pagination_data.total_count_with_limit
if limited_total_count > Kaminari::ActiveRecordRelationMethods::MAX_COUNT_LIMIT
pagination_data.without_count
else
pagination_data
end
end
2018-11-08 19:23:39 +05:30
def add_default_order(relation)
if relation.is_a?(ActiveRecord::Relation) && relation.order_values.empty?
2019-07-07 11:18:12 +05:30
relation = relation.order(:id) # rubocop: disable CodeReuse/ActiveRecord
2018-11-08 19:23:39 +05:30
end
relation
end
def add_pagination_headers(paginated_data)
header 'X-Per-Page', paginated_data.limit_value.to_s
header 'X-Page', paginated_data.current_page.to_s
header 'X-Next-Page', paginated_data.next_page.to_s
header 'X-Prev-Page', paginated_data.prev_page.to_s
header 'Link', pagination_links(paginated_data)
return if data_without_counts?(paginated_data)
header 'X-Total', paginated_data.total_count.to_s
header 'X-Total-Pages', total_pages(paginated_data).to_s
end
def pagination_links(paginated_data)
2019-07-07 11:18:12 +05:30
[].tap do |links|
links << %(<#{page_href(page: paginated_data.prev_page)}>; rel="prev") if paginated_data.prev_page
links << %(<#{page_href(page: paginated_data.next_page)}>; rel="next") if paginated_data.next_page
links << %(<#{page_href(page: 1)}>; rel="first")
2018-11-08 19:23:39 +05:30
2019-07-07 11:18:12 +05:30
links << %(<#{page_href(page: total_pages(paginated_data))}>; rel="last") unless data_without_counts?(paginated_data)
end.join(', ')
2018-11-08 19:23:39 +05:30
end
def total_pages(paginated_data)
# Ensure there is in total at least 1 page
[paginated_data.total_pages, 1].max
end
def data_without_counts?(paginated_data)
paginated_data.is_a?(Kaminari::PaginatableWithoutCount)
end
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
end
end
end