Rails3 アプリのスコープに基づいてアソシエーションを熱心にロードしようとしましたが、解決策が見つかりませんでした。
私のアプリには次のモデルがあります:
class Project
has_many :entries
has_many :to_dos
class ToDo
has_may :entries
has_many :tasks
belongs_to :project
class Task
has_many :entries
belongs_to :to_do
class Entry
belongs_to :project
belongs_to :to_do
belongs_to :task
# options format: {:from_date=>(Date.today-1.week), :to_date=>(Date.today+1.week), :user_id=>60}
scope :filtered_list, lambda { |options|
condition = options[:user_id].nil? ? "true" : "user_id = #{options[:user_id]}"
condition += options[:from_date].nil? ? "" : " AND entry_date >= '#{options[:from_date]}'"
condition += options[:to_date].nil? ? "" : " AND entry_date <= '#{options[:to_date]}'"
where(condition)
}
また、projects#index には、ユーザーのすべてのプロジェクトを取得する次のコードがあります。
@projects = current_user.projects.includes(:entries, :to_dos =>[:entries, :tasks => :entries])
ユーザーのすべてのプロジェクトを取得し、関連付けを熱心に読み込みます。したがって、プロジェクト内のすべてのエントリを取得するために次のループを実行すると、新しいクエリは発生しません。
def all_entries(options)
entries = self.entries
self.to_dos.each do |d|
entries += d.entries
d.tasks.each do |t|
entries += t.entries
end
end
end
この熱心な読み込みはすべてのエントリを取得するため、実際に必要な量よりもデータが多すぎます。そのため、熱心にロードされたエントリにいくつかの条件を適用しようとしましたが、解決策が見つかりませんでした。私は次のようなものを探していました:
@projects = current_user.projects.includes(:entries.filtered_list(options), :to_dos =>[:entries.filtered_list(options), :tasks => :entries.filtered_list(options)])
いくつかの条件を満たすエントリのみが読み込まれるようにします。
熱心な読み込みでスコープを使用することはできませんか? スコーピングと一緒にeagerloadingを使用するのを手伝ってください。