Skip to content

Instantly share code, notes, and snippets.

@kyamaguchi
Forked from troy/oneline_detailed_logging.rb
Created December 7, 2012 05:55
Show Gist options
  • Save kyamaguchi/4231083 to your computer and use it in GitHub Desktop.
Save kyamaguchi/4231083 to your computer and use it in GitHub Desktop.
One-line detailed logging for Rails 3 (ActiveSupport::Notifications) and later
# Outputs this at warn log level:
# 1.2.3.4 GET /path 200 OK BlahController#action HTML 938.2 (DB 11.8, View 719.7) {params} {optional params from flash[:log]}
#
# Save as config/initializers/oneline_detailed_logging.rb. Consider
# decreasing the log level from "info" to "warn" (in production.rb) so
# the one-line log message replaces the standard request logs.
# override process_action to add 2 things to the payload:
# - remote IP
# - an optional stash which is available to subscribers. Basically, flash[:log].inspect will be logged.
# 3.2: https://github.com/rails/rails/blob/3-2-stable/actionpack/lib/action_controller/metal/instrumentation.rb
ActionController::Instrumentation.send(:define_method, "process_action") do |arg|
raw_payload = {
:controller => self.class.name,
:action => self.action_name,
:params => request.filtered_parameters,
:format => request.format.try(:ref),
:method => request.method,
:path => (request.fullpath rescue "unknown"),
:ip => request.remote_ip,
:stash => request.session['flash'] && request.session['flash'][:log]
}
ActiveSupport::Notifications.instrument("start_processing.action_controller", raw_payload.dup)
ActiveSupport::Notifications.instrument("process_action.action_controller", raw_payload) do |payload|
result = super(arg)
payload[:status] = response.status
append_info_to_payload(payload)
result
end
end
ActiveSupport::Notifications.subscribe "process_action.action_controller" do |name, start, finish, id, payload|
# borrows from
# https://github.com/rails/rails/blob/3-2-stable/actionpack/lib/action_controller/log_subscriber.rb
params = payload[:params].except('controller', 'action', 'format', '_method', 'only_path')
format = payload[:format]
format = format.to_s.upcase if format.is_a?(Symbol)
duration = (finish-start)*1000
status = payload[:status]
if status.nil? && payload[:exception].present?
exception_class_name = payload[:exception].first
status = ActionDispatch::ExceptionWrapper.status_code_for_exception(exception_class_name)
end
m = "[summary] %s %s %s %s %s %s\#%s %s %.1f (DB %.1f, View %.1f) %s %s" % [
payload[:ip], payload[:method], payload[:path],
status, Rack::Utils::HTTP_STATUS_CODES[status],
payload[:controller], payload[:action], format,
duration || 0, payload[:db_runtime] || 0, payload[:view_runtime] || 0,
params.inspect, payload[:stash].try(:inspect) || {}.inspect ]
Rails.logger.warn(m)
end
require "action_view/log_subscriber"
module ActionView
class LogSubscriber
def info(msg)
debug(msg)
end
def render_partial(event)
# Silencing
end
def render_collection(event)
# Silencing
end
end
end
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment