module CounterCulture module Extensions extend ActiveSupport::Concern module ClassMethods # this holds all configuration data def after_commit_counter_cache config = @after_commit_counter_cache || [] if superclass.respond_to?(:after_commit_counter_cache) && superclass.after_commit_counter_cache config = superclass.after_commit_counter_cache + config end config end # called to configure counter caches def counter_culture(relation, options = {}) unless @after_commit_counter_cache include AfterCommitAction unless include?(AfterCommitAction) # initialize callbacks only once after_create :_update_counts_after_create before_destroy :_update_counts_after_destroy, if: -> (model) do if model.respond_to?(:paranoia_destroyed?) !model.paranoia_destroyed? else true end end after_update :_update_counts_after_update if respond_to?(:before_restore) before_restore :_update_counts_after_create, if: -> (model) { model.deleted? } end # we keep a list of all counter caches we must maintain @after_commit_counter_cache = [] end if options[:column_names] && !options[:column_names].is_a?(Hash) raise ":column_names must be a Hash of conditions and column names" end # add the counter to our collection @after_commit_counter_cache << Counter.new(self, relation, options) end # checks all of the declared counter caches on this class for correctnes based # on original data; if the counter cache is incorrect, sets it to the correct # count # # options: # { :exclude => list of relations to skip when fixing counts, # :only => only these relations will have their counts fixed } # returns: a list of fixed record as an array of hashes of the form: # { :entity => which model the count was fixed on, # :id => the id of the model that had the incorrect count, # :what => which column contained the incorrect count, # :wrong => the previously saved, incorrect count, # :right => the newly fixed, correct count } # def counter_culture_fix_counts(options = {}) raise "No counter cache defined on #{name}" unless @after_commit_counter_cache options[:exclude] = Array(options[:exclude]) if options[:exclude] options[:exclude] = options[:exclude].try(:map) {|x| x.is_a?(Enumerable) ? x : [x] } options[:only] = [options[:only]] if options[:only] && !options[:only].is_a?(Enumerable) options[:only] = options[:only].try(:map) {|x| x.is_a?(Enumerable) ? x : [x] } @after_commit_counter_cache.flat_map do |counter| next if options[:exclude] && options[:exclude].include?(counter.relation) next if options[:only] && !options[:only].include?(counter.relation) reconciler = CounterCulture::Reconciler.new(counter, options.slice(:skip_unsupported, :batch_size)) reconciler.reconcile! reconciler.changes end.compact end end private # need to make sure counter_culture is only activated once # per commit; otherwise, if we do an update in an after_create, # we would be triggered twice within the same transaction -- once # for the create, once for the update def _wrap_in_counter_culture_active(&block) if @_counter_culture_active # don't do anything; we are already active for this transaction else block.call execute_after_commit { @_counter_culture_active = false} end end # called by after_create callback def _update_counts_after_create _wrap_in_counter_culture_active do @_counter_culture_active = true self.class.after_commit_counter_cache.each do |counter| # increment counter cache counter.change_counter_cache(self, :increment => true) end end end # called by after_destroy callback def _update_counts_after_destroy _wrap_in_counter_culture_active do @_counter_culture_active = true self.class.after_commit_counter_cache.each do |counter| # decrement counter cache counter.change_counter_cache(self, :increment => false) end end end # called by after_update callback def _update_counts_after_update _wrap_in_counter_culture_active do self.class.after_commit_counter_cache.each do |counter| # figure out whether the applicable counter cache changed (this can happen # with dynamic column names) counter_cache_name_was = counter.counter_cache_name_for(counter.previous_model(self)) counter_cache_name = counter.counter_cache_name_for(self) if counter.first_level_relation_changed?(self) || (counter.delta_column && counter.attribute_changed?(self, counter.delta_column)) || counter_cache_name != counter_cache_name_was @_counter_culture_active = true # increment the counter cache of the new value counter.change_counter_cache(self, :increment => true, :counter_column => counter_cache_name) # decrement the counter cache of the old value counter.change_counter_cache(self, :increment => false, :was => true, :counter_column => counter_cache_name_was) end end end end end end