# encoding: utf-8 require 'ting_yun/instrumentation/support/queue_time' require 'ting_yun/agent/transaction/transaction_state' require 'ting_yun/instrumentation/support/evented_subscriber' require 'ting_yun/agent/transaction' module TingYun module Instrumentation module Rails4 class ActionControllerSubscriber < TingYun::Instrumentation::Support::EventedSubscriber def start(name, id, payload) #THREAD_LOCAL_ACCES state = TingYun::Agent::TransactionState.tl_get request = state.request event = ControllerEvent.new(name, Time.now, nil, id, payload, request) push_event(event) # if state.execution_traced? start_transaction(state, event) # end rescue => e log_notification_error(e, name, 'start') end def finish(name, id, payload) #THREAD_LOCAL_ACCESS event = pop_event(id) event.payload.merge!(payload) state = TingYun::Agent::TransactionState.tl_get stop_transaction(state) rescue => e log_notification_error(e, name, 'finish') end def start_transaction(state, event) TingYun::Agent::Transaction.start(state, :controller, :request => event.request, :filtered_params => event.payload[:params], :apdex_start_time => event.queue_start, :transaction_name => event.metric_name) end def stop_transaction(state) # txn = state.current_transaction TingYun::Agent::Transaction.stop(state) end end class ControllerEvent < TingYun::Instrumentation::Support::Event attr_accessor :parent attr_reader :queue_start, :request def initialize(name, start, ending, transaction_id, payload, request) # We have a different initialize parameter list, so be explicit super(name, start, ending, transaction_id, payload, nil) @request = request @controller_class = payload[:controller].split('::') \ .inject(Object) { |m, o| m.const_get(o) } if request && request.respond_to?(:env) @queue_start = TingYun::Instrumentation::Support::QueueTime.parse_frontend_timestamp(request.env, self.time) end end def metric_name if TingYun::Agent.config[:'nbs.auto_action_naming'] @metric_name ||= "WebAction/Rails/#{metric_path}%2F#{metric_action}" else path end end def metric_path @controller_class.controller_path end def metric_class payload[:controller] end def metric_action payload[:action] end def path payload[:path] end def to_s "#" end end end end end TingYun::Support::LibraryDetection.defer do @name = :rails4_controller depends_on do defined?(::Rails) && ::Rails::VERSION::MAJOR.to_i == 4 end depends_on do defined?(ActionController) && defined?(ActionController::Base) end executes do ::TingYun::Agent.logger.info 'Installing Rails 4 Controller instrumentation' end executes do ::TingYun::Instrumentation::Rails4::ActionControllerSubscriber \ .subscribe(/^process_action.action_controller$/) end end