require "active_job" module RailsSemanticLogger module ActiveJob class LogSubscriber < ::ActiveSupport::LogSubscriber def enqueue(event) ex = event.payload[:exception_object] if ex log_with_formatter level: :error, event: event do |fmt| { message: "Failed enqueuing #{fmt.job_info} (#{ex.class} (#{ex.message})", exception: ex } end elsif event.payload[:aborted] log_with_formatter level: :info, event: event do |fmt| {message: "Failed enqueuing #{fmt.job_info}, a before_enqueue callback halted the enqueuing execution."} end else log_with_formatter event: event do |fmt| {message: "Enqueued #{fmt.job_info}"} end end end def enqueue_at(event) ex = event.payload[:exception_object] if ex log_with_formatter level: :error, event: event do |fmt| { message: "Failed enqueuing #{fmt.job_info} (#{ex.class} (#{ex.message})", exception: ex } end elsif event.payload[:aborted] log_with_formatter level: :info, event: event do |fmt| {message: "Failed enqueuing #{fmt.job_info}, a before_enqueue callback halted the enqueuing execution."} end else log_with_formatter event: event do |fmt| {message: "Enqueued #{fmt.job_info} at #{fmt.scheduled_at}"} end end end def perform_start(event) log_with_formatter event: event do |fmt| {message: "Performing #{fmt.job_info}"} end end def perform(event) ex = event.payload[:exception_object] if ex log_with_formatter event: event, log_duration: true, level: :error do |fmt| { message: "Error performing #{fmt.job_info} in #{event.duration.round(2)}ms", exception: ex } end else log_with_formatter event: event, log_duration: true do |fmt| {message: "Performed #{fmt.job_info} in #{event.duration.round(2)}ms"} end end end private class EventFormatter def initialize(event:, log_duration: false) @event = event @log_duration = log_duration end def job_info "#{job.class.name} (Job ID: #{job.job_id}) to #{queue_name}" end def payload {}.tap do |h| h[:event_name] = event.name h[:adapter] = adapter_name h[:queue] = job.queue_name h[:job_class] = job.class.name h[:job_id] = job.job_id h[:provider_job_id] = job.try(:provider_job_id) # Not available in Rails 4.2 h[:duration] = event.duration.round(2) if log_duration? h[:arguments] = formatted_args end end def queue_name adapter_name + "(#{job.queue_name})" end def scheduled_at Time.at(event.payload[:job].scheduled_at).utc end private attr_reader :event def job event.payload[:job] end def adapter_name event.payload[:adapter].class.name.demodulize.remove("Adapter") end def formatted_args if defined?(job.class.log_arguments?) && !job.class.log_arguments? "" else JSON.pretty_generate(job.arguments.map { |arg| format(arg) }) end end def format(arg) case arg when Hash arg.transform_values { |value| format(value) } when Array arg.map { |value| format(value) } when GlobalID::Identification begin arg.to_global_id rescue StandardError arg end else arg end end def log_duration? @log_duration end end def log_with_formatter(level: :info, **kw_args) fmt = EventFormatter.new(**kw_args) msg = yield fmt logger.public_send(level, **msg, payload: fmt.payload) end def logger ::ActiveJob::Base.logger end end end end