require 'erb' require 'yaml' require 'uri' module Appsignal class Config DEFAULT_CONFIG = { :debug => false, :ignore_errors => [], :ignore_actions => [], :filter_parameters => [], :send_params => true, :endpoint => 'https://push.appsignal.com', :instrument_net_http => true, :instrument_redis => true, :instrument_sequel => true, :skip_session_data => false, :enable_frontend_error_catching => false, :frontend_error_catching_path => '/appsignal_error_catcher', :enable_allocation_tracking => true, :enable_gc_instrumentation => false, :running_in_container => false, :enable_host_metrics => true, :enable_minutely_probes => false, :hostname => Socket.gethostname }.freeze ENV_TO_KEY_MAPPING = { 'APPSIGNAL_ACTIVE' => :active, 'APPSIGNAL_PUSH_API_KEY' => :push_api_key, 'APPSIGNAL_APP_NAME' => :name, 'APPSIGNAL_PUSH_API_ENDPOINT' => :endpoint, 'APPSIGNAL_FRONTEND_ERROR_CATCHING_PATH' => :frontend_error_catching_path, 'APPSIGNAL_DEBUG' => :debug, 'APPSIGNAL_LOG_PATH' => :log_path, 'APPSIGNAL_INSTRUMENT_NET_HTTP' => :instrument_net_http, 'APPSIGNAL_INSTRUMENT_REDIS' => :instrument_redis, 'APPSIGNAL_INSTRUMENT_SEQUEL' => :instrument_sequel, 'APPSIGNAL_SKIP_SESSION_DATA' => :skip_session_data, 'APPSIGNAL_ENABLE_FRONTEND_ERROR_CATCHING' => :enable_frontend_error_catching, 'APPSIGNAL_IGNORE_ERRORS' => :ignore_errors, 'APPSIGNAL_IGNORE_ACTIONS' => :ignore_actions, 'APPSIGNAL_FILTER_PARAMETERS' => :filter_parameters, 'APPSIGNAL_SEND_PARAMS' => :send_params, 'APPSIGNAL_HTTP_PROXY' => :http_proxy, 'APPSIGNAL_ENABLE_ALLOCATION_TRACKING' => :enable_allocation_tracking, 'APPSIGNAL_ENABLE_GC_INSTRUMENTATION' => :enable_gc_instrumentation, 'APPSIGNAL_RUNNING_IN_CONTAINER' => :running_in_container, 'APPSIGNAL_WORKING_DIR_PATH' => :working_dir_path, 'APPSIGNAL_ENABLE_HOST_METRICS' => :enable_host_metrics, 'APPSIGNAL_ENABLE_MINUTELY_PROBES' => :enable_minutely_probes, 'APPSIGNAL_HOSTNAME' => :hostname }.freeze attr_reader :root_path, :env, :initial_config, :config_hash attr_accessor :logger def initialize(root_path, env, initial_config={}, logger=Appsignal.logger) @root_path = root_path @env = env.to_s @initial_config = initial_config @logger = logger @valid = false # Initial config @config_hash = DEFAULT_CONFIG.merge(initial_config) # Load config from environment variables load_from_environment # Load the config file if it exists if config_file && File.exists?(config_file) load_from_disk end # Validate that we have a correct config validate end def [](key) config_hash[key] end def []=(key, value) config_hash[key] = value end def log_file_path path = config_hash[:log_path] || root_path if path && File.writable?(path) File.join(File.realpath(path), 'appsignal.log') else '/tmp/appsignal.log' end rescue Errno::ENOENT '/tmp/appsignal.log' end def valid? @valid end def active? @valid && self[:active] end def write_to_environment ENV['APPSIGNAL_ACTIVE'] = active?.to_s ENV['APPSIGNAL_APP_PATH'] = root_path.to_s ENV['APPSIGNAL_AGENT_PATH'] = File.expand_path('../../../ext', __FILE__).to_s ENV['APPSIGNAL_ENVIRONMENT'] = env ENV['APPSIGNAL_AGENT_VERSION'] = Appsignal::Extension.agent_version ENV['APPSIGNAL_LANGUAGE_INTEGRATION_VERSION'] = Appsignal::VERSION ENV['APPSIGNAL_DEBUG_LOGGING'] = config_hash[:debug].to_s ENV['APPSIGNAL_LOG_FILE_PATH'] = log_file_path.to_s if log_file_path ENV['APPSIGNAL_PUSH_API_ENDPOINT'] = config_hash[:endpoint] ENV['APPSIGNAL_PUSH_API_KEY'] = config_hash[:push_api_key] ENV['APPSIGNAL_APP_NAME'] = config_hash[:name] ENV['APPSIGNAL_HTTP_PROXY'] = config_hash[:http_proxy] ENV['APPSIGNAL_IGNORE_ACTIONS'] = config_hash[:ignore_actions].join(',') ENV['APPSIGNAL_FILTER_PARAMETERS'] = config_hash[:filter_parameters].join(',') ENV['APPSIGNAL_SEND_PARAMS'] = config_hash[:send_params].to_s ENV['APPSIGNAL_RUNNING_IN_CONTAINER'] = config_hash[:running_in_container].to_s ENV['APPSIGNAL_WORKING_DIR_PATH'] = config_hash[:working_dir_path] if config_hash[:working_dir_path] ENV['APPSIGNAL_ENABLE_HOST_METRICS'] = config_hash[:enable_host_metrics].to_s ENV['APPSIGNAL_ENABLE_MINUTELY_PROBES'] = config_hash[:enable_minutely_probes].to_s ENV['APPSIGNAL_HOSTNAME'] = config_hash[:hostname].to_s ENV['APPSIGNAL_PROCESS_NAME'] = $0 end protected def config_file @config_file ||= root_path.nil? ? nil : File.join(root_path, 'config', 'appsignal.yml') end def load_from_disk configurations = YAML.load(ERB.new(IO.read(config_file)).result) config_for_this_env = configurations[env] if config_for_this_env config_for_this_env = Hash[config_for_this_env.map do |key, value| [key.to_sym, value] end] # convert keys to symbols # Backwards compatibility with config files generated by earlier # versions of the gem if !config_for_this_env[:push_api_key] && config_for_this_env[:api_key] config_for_this_env[:push_api_key] = config_for_this_env[:api_key] end if !config_for_this_env[:ignore_errors] && config_for_this_env[:ignore_exceptions] config_for_this_env[:ignore_errors] = config_for_this_env[:ignore_exceptions] end merge(@config_hash, config_for_this_env) else @logger.error "Not loading from config file: config for '#{env}' not found" end end def load_from_environment config = {} # Make active by default if APPSIGNAL_PUSH_API_KEY is present if ENV['APPSIGNAL_PUSH_API_KEY'] config[:active] = true end # Heroku is a container based system if ENV['DYNO'] config[:running_in_container] = true end # Configuration with string type %w(APPSIGNAL_PUSH_API_KEY APPSIGNAL_APP_NAME APPSIGNAL_PUSH_API_ENDPOINT APPSIGNAL_FRONTEND_ERROR_CATCHING_PATH APPSIGNAL_HTTP_PROXY APPSIGNAL_LOG_PATH APPSIGNAL_WORKING_DIR_PATH APPSIGNAL_HOSTNAME).each do |var| if env_var = ENV[var] config[ENV_TO_KEY_MAPPING[var]] = env_var end end # Configuration with boolean type %w(APPSIGNAL_ACTIVE APPSIGNAL_DEBUG APPSIGNAL_INSTRUMENT_NET_HTTP APPSIGNAL_SKIP_SESSION_DATA APPSIGNAL_ENABLE_FRONTEND_ERROR_CATCHING APPSIGNAL_ENABLE_ALLOCATION_TRACKING APPSIGNAL_ENABLE_GC_INSTRUMENTATION APPSIGNAL_RUNNING_IN_CONTAINER APPSIGNAL_ENABLE_HOST_METRICS APPSIGNAL_SEND_PARAMS APPSIGNAL_ENABLE_MINUTELY_PROBES).each do |var| if env_var = ENV[var] config[ENV_TO_KEY_MAPPING[var]] = env_var == 'true' end end # Configuration with array of strings type %w(APPSIGNAL_IGNORE_ERRORS APPSIGNAL_IGNORE_ACTIONS APPSIGNAL_FILTER_PARAMETERS).each do |var| if env_var = ENV[var] config[ENV_TO_KEY_MAPPING[var]] = env_var.split(',') end end merge(@config_hash, config) end def merge(original_config, new_config) new_config.each do |key, value| unless original_config[key].nil? @logger.debug("Config key '#{key}' is being overwritten") end original_config[key] = value end end def validate # Strip path from endpoint so we're backwards compatible with # earlier versions of the gem. endpoint_uri = URI(config_hash[:endpoint]) if endpoint_uri.port == 443 config_hash[:endpoint] = "#{endpoint_uri.scheme}://#{endpoint_uri.host}" else config_hash[:endpoint] = "#{endpoint_uri.scheme}://#{endpoint_uri.host}:#{endpoint_uri.port}" end if config_hash[:push_api_key] @valid = true else @valid = false @logger.error "Push api key not set after loading config" end end def load_default_config_with_push_api_key_and_name_from_env(key) # Get base config by doing the default merge and adding the push api key. @config_hash = merge_config( :push_api_key => key, :active => true ) @config_hash[:name] = ENV['APPSIGNAL_APP_NAME'] if ENV['APPSIGNAL_APP_NAME'] @config_hash[:active] = ENV['APPSIGNAL_ACTIVE'] == 'true' if ENV['APPSIGNAL_ACTIVE'] end end end