10

有什么方法可以在每次 ActiveRecord 回调发生时自动记录?当记录有多个回调到位时,这将有助于追溯为什么会发生某些事情。

我希望看到自动日志消息,指示正在调用哪些消息以响应哪些回调,例如:before_validation: calling update_capitalization

4

4 回答 4

9

我只是为此写了一个宝石。

# Gemfile
gem "rails-callback_log", group: [:development, :test]
于 2016-06-24T22:59:26.400 回答
3

对于谷歌和后代(在 Rails 3 上):

module CallbackTrace
  def self.included kls
    kls.send :alias_method_chain, :_compile_filter, :trace
  end

  def _compile_filter_with_trace filter
    generated_code = _compile_filter_without_trace(filter)
    return generated_code if filter.is_a?(Array)
    method_name = generated_code.to_s.split(%r{\(|\s}).first
    _klass = @klass
    prelogger = ->{
        Rails.logger.info("START [#{filter.class}](#{generated_code})")
        Rails.logger.info("#{_klass} #{Time.now}")
        if imethod=(_klass.instance_method(method_name) rescue nil)
          begin
            Rails.logger.info(imethod.source)
          rescue MethodSource::SourceNotFoundError
            Rails.logger.info("NO SOURCE FOR #{generated_code}")
          end
        else
          Rails.logger.info("NO METHOD: #{method_name} for #{@klass}")
        end
    }
    postlogger = ->{
      Rails.logger.info("ENDED #{generated_code} #{Time.now}")
    }
    @klass.send :define_method, "prelogger_#{method_name}", &prelogger
    @klass.send :define_method, "postlogger_#{method_name}", &postlogger
    "(prelogger_#{method_name}; retval = retval = #{generated_code}; " +
        "postlogger_#{method_name}; retval)"
  end
end
ActiveSupport::Callbacks::Callback.send :include, CallbackTrace
于 2013-07-16T04:46:50.470 回答
0

更简单的方法是使用 rails 本机记录器方法:

调试 Rails 应用程序

像这样使用它:

logger.debug "Person attributes hash: #{@person.attributes.inspect}"
logger.info "Processing the request..."
logger.fatal "Terminating application, raised unrecoverable error!!!"

编辑:

检查发生异常时调用的内容也很有趣

logger.error("trace: #{ex.backtrace().join("\n")}")
于 2012-10-26T16:26:04.720 回答
0

另一种方法是在所有要记录的模型上定义回调。为避免加载所有内容,您只需加载模型即可。为简单起见,这会尝试加载与app/models/*.rb. 我也会使用puts,但你可以使用Rails.logger.info或其他。对于临时调试,您可以将其放入初始化程序中,例如config/initializers/001_log_callbacks.rb

# Log all callbacks
Dir[Rails.root.join('app/models/*.rb').to_s].each do |filename|
  name = File.basename(filename, '.rb')
  begin
    model = name.camelize.constantize
    ActiveRecord::Callbacks::CALLBACKS.each do |callback|
      if callback.to_s.start_with?('around_')
        model.class_eval "#{callback} do |*args, &prc|; puts \"#{model}(\#{id}).#{callback} start\"; prc.call(*args) if prc; puts \"#{model}(\#{id}).#{callback} end\" end"
      else
        model.class_eval "#{callback} do; puts \"#{model}(\#{id}).#{callback}\" end"
      end
    end
  rescue
  end
end
于 2014-03-07T19:30:52.863 回答