# encoding: utf-8
# This file is distributed under Ting Yun's license terms.
require 'ting_yun/frameworks'
require 'ting_yun/version'
# before the real start,do check and log things
module TingYun
module Agent
module InstanceMethods
module Start
# Check to see if the agent should start, returning +true+ if it should.
# should hava the vaild app_name, unstart-state and able to start
def agent_should_start?
return false if already_started? || disabled?
unless app_name_configured?
TingYun::Agent.logger.error "No application name configured.",
"The Agent cannot start without at least one. Please check your ",
"tingyun.yml and ensure that it is valid and has at least one ",
"value set for app_name in the",
"environment."
return false
end
return true
end
def started?
@started
end
# Check whether we have already started, which is an error condition
def already_started?
if started?
TingYun::Agent.logger.info("Agent Started Already!")
true
end
end
# The agent is disabled when it is not force enabled by the
# 'nbs.agent_enabled' option (e.g. in a manual start), or
# enabled normally through the configuration file
def disabled?
!TingYun::Agent.config[:'nbs.agent_enabled']
end
def log_startup
log_environment
log_dispatcher
log_app_name
end
def log_environment
Agent.logger.info "Environment: #{::TingYun::Frameworks.framework.env}"
end
# Logs the dispatcher to the log file to assist with
# debugging. When no debugger is present, logs this fact to
# assist with proper dispatcher detection
def log_dispatcher
dispatcher_name = TingYun::Agent.config[:dispatcher].to_s
if dispatcher_name.empty?
TingYun::Agent.logger.info 'No known dispatcher detected.'
else
TingYun::Agent.logger.info "Dispatcher: #{dispatcher_name}"
end
end
def log_app_name
TingYun::Agent.logger.info "Application: #{TingYun::Agent.config.app_names.join(", ")}"
end
def sinatra_app?
(
defined?(Sinatra::Application) &&
Sinatra::Application.respond_to?(:run) &&
Sinatra::Application.run?
)
end
# Classy logging of the agent version and the current pid,
# so we can disambiguate processes in the log file and make
# sure they're running a reasonable version
def log_version_and_pid
TingYun::Agent.logger.debug "Ting Yun Ruby Agent #{TingYun::VERSION::STRING} Initialized: pid = #{$$}"
end
# Warn the user if they have configured their agent not to
# send data, that way we can see this clearly in the log file
def monitoring?
if TingYun::Agent.config[:monitor_mode]
true
else
TingYun::Agent.logger.warn('Agent configured not to send data in this environment.')
false
end
end
# Tell the user when the license key is missing so they can
# fix it by adding it to the file
def has_license_key?
if TingYun::Agent.config[:license_key] && TingYun::Agent.config[:license_key].length > 0
true
else
TingYun::Agent.logger.warn("No license key found. " +
"This often means your tingyun.yml file was not found, or it lacks a section for the running environment,'#{::TingYun::Frameworks.framework.env}'. You may also want to try linting your tingyun.yml to ensure it is valid YML.")
false
end
end
# A license key is an arbitrary 40 character string,
# usually looks something like a SHA1 hash
def correct_license_length
key = TingYun::Agent.config[:license_key]
if key.length > 0
true
else
TingYun::Agent.logger.error("Invalid license key: #{key}")
false
end
end
# A correct license key exists and is of the proper length
def has_correct_license_key?
has_license_key? && correct_license_length
end
# Logs the configured application names
def app_name_configured?
names = TingYun::Agent.config.app_names
return names.respond_to?(:any?) && names.any?
end
# If we're using a dispatcher that forks before serving
# requests, we need to wait until the children are forked
# before connecting, otherwise the parent process sends useless data
def is_using_forking_dispatcher?
if [:puma, :passenger, :rainbows, :unicorn].include? TingYun::Agent.config[:dispatcher]
TingYun::Agent.logger.info "Deferring startup of agent reporting thread because #{TingYun::Agent.config[:dispatcher]} may fork."
true
else
false
end
end
# Sanity-check the agent configuration and start the agent,
# setting up the worker thread and the exit handler to shut
# down the agent
def check_config_and_start_agent
return unless monitoring? && has_correct_license_key?
return if is_using_forking_dispatcher?
setup_and_start_agent
end
# This is the shared method between the main agent startup and the
# after_fork call restarting the thread in deferred dispatchers.
#
# Treatment of @started and env report is important to get right.
def setup_and_start_agent(options={})
@started = true
@dispatcher.mark_started
generate_environment_report
install_exit_handler
cpu_and_memory
start_worker_thread(options)
end
# This method should be called in a forked process after a fork.
# It assumes the parent process initialized the agent, but does
# not assume the agent started.
#
# The call is idempotent, but not re-entrant.
#
# * It clears any metrics carried over from the parent process
# * Restarts the sampler thread if necessary
# * Initiates a new agent run and worker loop unless that was done
# in the parent process and +:force_reconnect+ is not true
#
# Options:
# * :force_reconnect => true to force the spawned process to
# establish a new connection, such as when forking a long running process.
# The default is false--it will only connect to the server if the parent
# had not connected.
# * :keep_retrying => false if we try to initiate a new
# connection, this tells me to only try it once so this method returns
# quickly if there is some kind of latency with the server.
def after_fork(options={})
needs_restart = false
@after_fork_lock.synchronize do
needs_restart = @dispatcher.needs_restart?
@dispatcher.mark_started
end
return if !needs_restart ||
!Agent.config[:'nbs.agent_enabled'] ||
!Agent.config[:monitor_mode] ||
disconnected?
::TingYun::Agent.logger.debug "Starting the worker thread in #{Process.pid} (parent #{Process.ppid}) after forking."
# Clear out locks and stats left over from parent process
reset_objects_with_locks
drop_buffered_data
setup_and_start_agent(options)
end
def cpu_and_memory
@middleware.load_samplers
end
end
end
end
end