require 'roqua/support' require 'active_support/log_subscriber' module Roqua module Support module RequestLogging def add_log_information(key, value) Thread.current[:roqua_request_log] ||= {} Thread.current[:roqua_request_log][key] = value end end class RequestLogger < ActiveSupport::LogSubscriber include Roqua::Support::Instrumentation def process_action(event) payload = event.payload extra_logged_information = Thread.current[:roqua_request_log] || {} Thread.current[:roqua_request_log] = {} data = extract_request_id(event) data.merge! extract_request(payload) data.merge! extract_status(payload) data.merge! extract_parameters(payload) data.merge! redirect_information data.merge! extra_logged_information data.merge! runtimes(event) #eventlog.info event.inspect eventlog.info "web", data rescue Exception => e eventlog.info "web:logerror", {class: e.class, message: e.message} raise end def redirect_to(event) # Unfortunately, when a redirect is triggered by your application's code, # ActionController fires two events. One for the redirect itself, and # another one when the request is finished. Unfortunately the final event # doesn't include the redirect, so we store the redirect URL as a # thread-local attribute and refers to it in process_action. Thread.current[:roqua_request_log_redirect] = event.payload[:location] end private def extract_request_id(event) {uuid: event.transaction_id} end def extract_request(payload) { :method => payload[:method], :path => extract_path(payload), :format => extract_format(payload), :controller => payload[:params]['controller'], :action => payload[:params]['action'] } end def extract_path(payload) payload[:path].split("?").first end def extract_format(payload) payload[:format] end def extract_status(payload) if payload[:status] { :status => payload[:status].to_i } elsif payload[:exception] exception, message = payload[:exception] { :status => 500, :error => "#{exception}:#{message}" } else { :status => 0 } end end def redirect_information if location = Thread.current[:roqua_request_log_redirect] Thread.current[:roqua_request_log_redirect] = nil {location: location} else {} end end def extract_parameters(payload) filtered_params = payload[:params].reject do |key, value| key == 'controller' or key == 'action' end {params: filtered_params} end def runtimes(event) { :duration => event.duration, :view => event.payload[:view_runtime], :db => event.payload[:db_runtime] }.inject({}) do |runtimes, (name, runtime)| runtimes[name] = runtime.to_f.round(2) if runtime runtimes end end end end end