Kaminari pagination is not supported by Tapioca
lavoiesl opened this issue · 1 comments
Trying to update tapioca from 0.12.0 to 0.13.1 results in this error:
path/to/file.rb:32: Method page does not exist on Model::PrivateAssociationRelation https://srb.help/7003
32 | @records = Model.all.page(params[:page]).per(25)
Investigation
The page method is defined on the class by kaminari:
https://github.com/kaminari/kaminari/blob/9182d065c144afa45c6b7cf444f810bea1fd7201/kaminari-activerecord/lib/kaminari/activerecord/active_record_model_extension.rb#L14-L23
But attempting to introspect it at runtime yields nothing:
pry(main)> Model.all.method(:page)
=> #<Method: Model::ActiveRecord_Relation#page(*)>
pry(main)> Model.all.method(:page).source_location
=> nil
pry(main)> Model.all.method(:page).owner
=> Model::ActiveRecord_Relation
pry(main)> Model.all.method(:page).owner.instance_method(:page)
NameError: undefined method `page' for class `ActiveRecord::Relation'
from (pry):1:in `instance_method'from (pry):35:in `method'
pry(main)> Model.all.methods.include?(:page)
=> false
This is because it is delegated by https://github.com/rails/rails/blob/d37c533139f70efdcd95f8dadd48c10eba429f94/activerecord/lib/active_record/relation/delegation.rb#L71-L88
Which is generated on demand when the method is missing:
https://github.com/rails/rails/blob/d37c533139f70efdcd95f8dadd48c10eba429f94/activerecord/lib/active_record/relation/delegation.rb#L115-L124
Attempting the same as above after calling the method once does yield something a bit more interesting:
pry(main)> Model.all.method(:page)
=> #<Method: Model::ActiveRecord_Relation(Model::GeneratedRelationMethods)#page(...) /Users/seb/.gem/ruby/3.2.2/gems/activerecord-7.1.3.2/lib/active_record/relation/delegation.rb:78>
pry(main)> Model.all.method(:page).source_location
=> ["/Users/seb/.gem/ruby/3.2.2/gems/activerecord-7.1.3.2/lib/active_record/relation/delegation.rb", 78]
pry(main)> Model.all.method(:page).owner
=> Model::GeneratedRelationMethods
pry(main)> Model.all.method(:page).owner.instance_method(:page)
=> #<UnboundMethod: Model::GeneratedRelationMethods#page(...) /Users/seb/.gem/ruby/3.2.2/gems/activerecord-7.1.3.2/lib/active_record/relation/delegation.rb:78>
pry(main)> Model.all.methods.include?(:page)
=> true
The same can be achieved by calling Model.generate_relation_method(:page)
Workaround
Adding this in config/initializers/kaminari.rb
makes tapioca generate the methods:
if defined?(Tapioca)
Rails.application.config.after_initialize do
Rails.application.eager_load!
ActiveRecord::Base.descendants.each do |model|
model.generate_relation_method(:page)
model.generate_relation_method(:per)
end
end
end
The solution above works pretty well, but it's not aware of methods like #total_count
, etc. Here's a compiler that makes the types a bit more accurate. It's based on the ActiveRecordScope
compiler, but I took some shortcuts.
I might get around to upstreaming this, but it'd be really cool if someone took it from here.
# typed: ignore
require 'tapioca/dsl/helpers/active_record_constants_helper'
require 'kaminari/activerecord/active_record_model_extension'
module Tapioca
module Dsl
module Compilers
class Kaminari < Tapioca::Dsl::Compiler
include Tapioca::Dsl::Helpers::ActiveRecordConstantsHelper
def self.gather_constants
all_classes
.select { |c| c < ::Kaminari::ActiveRecordModelExtension }
.reject(&:abstract_class?)
end
def decorate
root.create_path(constant) do |model|
generate_page_method(
model.create_module(RelationMethodsModuleName),
RelationClassName,
)
generate_page_method(
model.create_module(AssociationRelationMethodsModuleName),
AssociationRelationClassName,
)
model.create_extend(RelationMethodsModuleName)
end
end
private
def generate_page_method(mod, return_type)
mod.create_method(
'page',
parameters: [create_opt_param('num', type: 'T.any(Integer, String)', default: 'nil')],
return_type: "T.all(#{return_type}, Kaminari::PageScopeMethods, Kaminari::ActiveRecordRelationMethods)",
)
end
end
end
end
end