require 'logger' require 'rack' require 'thread_safe' require 'securerandom' begin require 'active_support/notifications' rescue LoadError require 'vendor/active_support/notifications' end module Appsignal class << self attr_accessor :config, :logger, :agent, :in_memory_log def load_integrations require 'appsignal/integrations/delayed_job' require 'appsignal/integrations/passenger' require 'appsignal/integrations/unicorn' require 'appsignal/integrations/sidekiq' require 'appsignal/integrations/resque' require 'appsignal/integrations/sequel' end def load_instrumentations require 'appsignal/instrumentations/net_http' if config[:instrument_net_http] end def extensions @extensions ||= [] end def initialize_extensions Appsignal.logger.debug('Initializing extensions') extensions.each do |extension| Appsignal.logger.debug("Initializing #{extension}") extension.initializer end end def start if config if config[:debug] logger.level = Logger::DEBUG else logger.level = Logger::INFO end if config.active? logger.info("Starting AppSignal #{Appsignal::VERSION} on #{RUBY_VERSION}/#{RUBY_PLATFORM}") load_integrations load_instrumentations initialize_extensions @agent = Appsignal::Agent.new at_exit do logger.debug('Running at_exit block') @agent.send_queue end else logger.info("Not starting, not active for #{config.env}") end else logger.error('Can\'t start, no config loaded') end end # Convenience method for adding a transaction to the queue. This queue is # managed and is periodically pushed to Appsignal. # # @return [ true ] True. # # @since 0.5.0 def enqueue(transaction) return unless active? agent.enqueue(transaction) end def monitor_transaction(name, payload={}) unless active? yield return end begin Appsignal::Transaction.create(SecureRandom.uuid, ENV) ActiveSupport::Notifications.instrument(name, payload) do yield end rescue Exception => exception Appsignal.add_exception(exception) raise exception ensure Appsignal::Transaction.complete_current! end end def listen_for_exception(&block) yield rescue Exception => exception send_exception(exception) raise exception end def send_exception(exception, tags=nil) return if !active? || is_ignored_exception?(exception) transaction = Appsignal::Transaction.create(SecureRandom.uuid, ENV) transaction.add_exception(exception) transaction.set_tags(tags) if tags transaction.complete! Appsignal.agent.send_queue end def add_exception(exception) return if !active? || Appsignal::Transaction.current.nil? || exception.nil? || is_ignored_exception?(exception) Appsignal::Transaction.current.add_exception(exception) end def tag_request(params={}) return unless active? transaction = Appsignal::Transaction.current return false unless transaction transaction.set_tags(params) end alias :tag_job :tag_request def transactions @transactions ||= {} end def logger @in_memory_log = StringIO.new unless @in_memory_log @logger ||= Logger.new(@in_memory_log).tap do |l| l.level = Logger::INFO l.formatter = Logger::Formatter.new end end def start_logger(path) if path && File.writable?(path) && !ENV['DYNO'] && !ENV['SHELLYCLOUD_DEPLOYMENT'] @logger = Logger.new(File.join(path, 'appsignal.log')) @logger.formatter = Logger::Formatter.new else @logger = Logger.new($stdout) @logger.formatter = lambda do |severity, datetime, progname, msg| "appsignal: #{msg}\n" end end @logger.level = Logger::INFO @logger << @in_memory_log.string if @in_memory_log end def post_processing_middleware @post_processing_chain ||= Appsignal::Aggregator::PostProcessor.default_middleware yield @post_processing_chain if block_given? @post_processing_chain end def active? config && config.active? && agent && agent.active? end def is_ignored_exception?(exception) Appsignal.config[:ignore_exceptions].include?(exception.class.name) end def is_ignored_action?(action) Appsignal.config[:ignore_actions].include?(action) end # Convenience method for skipping instrumentations around a block of code. # # @since 0.8.7 def without_instrumentation Appsignal::Transaction.current.pause! if Appsignal::Transaction.current yield ensure Appsignal::Transaction.current.resume! if Appsignal::Transaction.current end end end require 'appsignal/agent' require 'appsignal/event' require 'appsignal/aggregator' require 'appsignal/aggregator/post_processor' require 'appsignal/aggregator/middleware' require 'appsignal/auth_check' require 'appsignal/config' require 'appsignal/marker' require 'appsignal/rack/listener' require 'appsignal/rack/instrumentation' require 'appsignal/rack/sinatra_instrumentation' require 'appsignal/rack/js_exception_catcher' require 'appsignal/params_sanitizer' require 'appsignal/transaction' require 'appsignal/transaction/formatter' require 'appsignal/transaction/params_sanitizer' require 'appsignal/transmitter' require 'appsignal/ipc' require 'appsignal/version' require 'appsignal/integrations/rails' require 'appsignal/js_exception_transaction'