|
module Arel |
|
module Nodes |
|
module Cursor |
|
class LessThan < Arel::Nodes::LessThan; end |
|
class GreaterThan < Arel::Nodes::GreaterThan; end |
|
end |
|
end |
|
|
|
module Predications |
|
def before right |
|
Nodes::Cursor::LessThan.new self, quoted_node(right) |
|
end |
|
|
|
def after right |
|
Nodes::Cursor::GreaterThan.new self, quoted_node(right) |
|
end |
|
end |
|
end |
|
|
|
module Paginatable |
|
extend ActiveSupport::Concern |
|
|
|
included do |
|
def self.all(*args) |
|
super.extending(ActiveRecordRelationMethods) |
|
end |
|
|
|
def self.after(cursor) |
|
reorder(primary_key => :asc).where( |
|
arel_table[primary_key].after(cursor) |
|
).extending(ActiveRecordRelationMethods) |
|
end |
|
|
|
def self.before(cursor) |
|
reorder(primary_key => :desc).where( |
|
arel_table[primary_key].before(cursor) |
|
).extending(ActiveRecordRelationMethods) |
|
end |
|
end |
|
|
|
module ActiveRecordRelationMethods |
|
def has_more? |
|
@has_more ||= begin |
|
# Cache #size otherwise multiple calls to the database will occur. |
|
(results_size = size) > 0 && results_size < total_size |
|
end |
|
end |
|
|
|
# Returns number of records that exist in scope of the current cursor |
|
def total_size(column_name = :all) #:nodoc: |
|
# #count overrides the #select which could include generated columns |
|
# referenced in #order, so skip #order here, where it's irrelevant to the |
|
# result anyway. |
|
@total_size ||= begin |
|
context = except(:offset, :limit, :order) |
|
|
|
# Remove includes only if they are irrelevant |
|
context = context.except(:includes) unless references_eager_loaded_tables? |
|
|
|
args = [column_name] |
|
|
|
# .group returns an OrderedHash that responds to #count |
|
context = context.count(*args) |
|
|
|
if context.is_a?(Hash) || context.is_a?(ActiveSupport::OrderedHash) |
|
context.count |
|
else |
|
context.respond_to?(:count) ? context.count(*args) : context |
|
end |
|
end |
|
end |
|
|
|
# Returns number of records that exist without :offset, :limit, :order, :before or :after |
|
def total_count(column_name = :all) #:nodoc: |
|
# #count overrides the #select which could include generated columns |
|
# referenced in #order, so skip #order here, where it's irrelevant to the |
|
# result anyway. |
|
@total_count ||= begin |
|
context = except(:offset, :limit, :order) |
|
|
|
context.where_values = where_values.reject do |value| |
|
value.is_a?(Arel::Nodes::Cursor::GreaterThan) || |
|
value.is_a?(Arel::Nodes::Cursor::LessThan) |
|
end |
|
|
|
# Remove includes only if they are irrelevant |
|
context = context.except(:includes) unless references_eager_loaded_tables? |
|
|
|
args = [column_name] |
|
|
|
# .group returns an OrderedHash that responds to #count |
|
context = context.count(*args) |
|
|
|
if context.is_a?(Hash) || context.is_a?(ActiveSupport::OrderedHash) |
|
context.count |
|
else |
|
context.respond_to?(:count) ? context.count(*args) : context |
|
end |
|
end |
|
end |
|
end |
|
end |
@jerrygreen
It's not a bad implementation or nonsense. It served my use case just fine. Just because it doesn't fit yours doesn't make it bad. I did not intend for it to account for sorting on anything other than the primary key. This is how most cursor based apis work. It doesn't break your sorting, it intentionally removes it. You can't sort by arbitrary columns. The field you compare the cursor to has to be unique, sequential and immutable.
It's been several years since I posted this. I think the solution I would implement now is to generate an opaque token for the cursor that contains the necessary information. One cursor for the next page and one for the previous would be sent back to the client.
Making it opaque to the client means you can change the implementation server side, which is a win.
Slack has a decent write up on this: https://slack.engineering/evolving-api-pagination-at-slack-1c1f644f8e12