# FIXME: files named with hyphens will not be found by Chamber for now module BmcDaemonLib # Class exceptions class ConfigInitiRequired < StandardError; end class ConfigMissingParameter < StandardError; end class ConfigOtherError < StandardError; end class ConfigParseError < StandardError; end class ConfigMultipleGemspec < StandardError; end class ConfigMissingGemspec < StandardError; end class Conf extend Chamber PIDFILE_DIR = "/tmp/" class << self attr_accessor :app_env attr_reader :app_root attr_reader :app_libs attr_reader :app_name attr_reader :app_ver attr_reader :app_started attr_reader :app_spec attr_reader :files attr_reader :host end def self.init app_root # Permanent flags @initialized = true @app_started = Time.now # Default values @files ||= [] @app_name ||= "app_name" @app_env ||= "production" @host ||= `hostname`.to_s.chomp.split(".").first # Store and clean app_root @app_root = File.expand_path(app_root) # Try to find any gemspec file matches = Dir["#{@app_root}/*.gemspec"] fail ConfigMissingGemspec, "gemspec file not found: #{gemspec_path}" if matches.size < 1 fail ConfigMultipleGemspec, "gemspec file not found: #{gemspec_path}" if matches.size > 1 # Load Gemspec (just the only match) @spec = Gem::Specification::load(matches.first) @app_name = @spec.name @app_ver = @spec.version fail ConfigMissingParameter, "gemspec: missing name" unless @app_name fail ConfigMissingParameter, "gemspec: missing version" unless @app_ver # Now we know app_name, initalize app_libs @app_libs = File.expand_path("lib/#{@app_name}/", @app_root) # By default, Newrelic is disabled ENV["NEWRELIC_AGENT_ENABLED"] = "false" # Add other config files add_config generate(:config_defaults) add_config generate(:config_etc) # Return something return @app_name end def self.prepare args = {} ensure_init # Add extra config file and load them all add_config args[:config] reload! # Set Rack env ENV["RACK_ENV"] = @app_env.to_s # Set up encodings Encoding.default_internal = "utf-8" Encoding.default_external = "utf-8" # Init New Relic, Rollbar # prepare_newrelic #prepare_rollbar if self.feature?(:rollbar) # Try to access any key to force parsing of the files self[:dummy] rescue Psych::SyntaxError => e fail ConfigParseError, e.message rescue StandardError => e fail ConfigOtherError, "#{e.message} \n #{e.backtrace.to_yaml}" end # Reload files def self.reload! ensure_init load_files end def self.dump ensure_init to_hash.to_yaml(indent: 4, useheader: true, useversion: false ) end # Direct access to any depth def self.at *path ensure_init path.reduce(Conf) { |m, key| m && m[key.to_s] } end def self.logfile_path pipe # Access configuration path = Conf.at :logs, :path specific = Conf.at :logs, pipe default = Conf.at :logs, :default # Ignore if explicitely disabled return nil if specific.nil? # Fallback on default path if not provided, specific ||= default specific ||= "default.log" # Build logfile_path File.expand_path specific.to_s, path.to_s end def self.feature? name ensure_init # Guess if the specific feature si available case name when :newrelic return false if Gem.datadir('newrelic_rpm').nil? return false if self.at(:newrelic, :enabled) == false return false if self.at(:newrelic, :disabled) == true return self.at(:newrelic, :license) || false when :rollbar return false if Gem.datadir('rollbar').nil? return false if self.at(:rollbar, :enabled) == false return false if self.at(:rollbar, :disabled) == true return self.at(:rollbar, :token) || false end return false end # Defaults generators def self.generate what ensure_init return case what when :user_agent "#{@app_name}/#{@app_ver}" if @app_name && @app_ver when :config_defaults "#{@app_root}/defaults.yml" if @app_root when :config_etc "/etc/#{@app_name}.yml" if @app_name when :process_name parts = [@app_name, @app_env] parts << self[:port] if self[:port] parts.join('-') when :pidfile process_name = self.generate(:process_name) File.expand_path "#{process_name}.pid", PIDFILE_DIR when :config_message config_defaults = self.generate(:config_defaults) config_etc = self.generate(:config_etc) "A default configuration is available (#{config_defaults}) and can be copied to the default location (#{config_etc}): \n sudo cp #{config_defaults} #{config_etc}" end end def self.prepare_newrelic # Disable if no config present return unless self.feature?(:newrelic) section = self[:newrelic] # Enable GC profiler GC::Profiler.enable # Set logfile, license, monitor mode ENV["NEW_RELIC_LOG"] = logfile_path(:newrelic) ENV["NEW_RELIC_LICENSE_KEY"] = section[:license].to_s ENV["NEW_RELIC_MONITOR_MODE"] = "true" # Build NewRelic app_name if not provided as-is if !section[:app_name] stack = [] stack << (section[:prefix] || @app_name) stack << section[:platform] if section[:platform] stack << @app_env text = stack.join('-') section[:app_name] = "#{text}; #{text}-#{host}" end ENV["NEW_RELIC_APP_NAME"] = section[:app_name].to_s # Enable module ENV["NEWRELIC_AGENT_ENABLED"] = "true" end def self.prepare_rollbar # Disable if no config present unless self.feature?(:rollbar) Rollbar.configure do |config| config.enabled = false end return end section = self[:rollbar] # Configure Rollbar.configure do |config| config.enabled = true config.access_token = section[:token].to_s config.code_version = @app_version config.environment = @app_env #config.logger = Logger.new(logfile_path(:rollbar)) config.logger = LoggerPool.instance.get :rollbar config.use_async = true end # Notify startup Rollbar.info("#{@app_name} #{@app_ver} [#{@host}]") end def self.log origin, message printf( "%s %-10s %s \n", Time.now.strftime("%Y-%m-%d %H:%M:%S"), origin, message ) end protected def self.load_files load files: @files, namespaces: { environment: @app_env } end def self.add_config path @files << File.expand_path(path) if path && File.readable?(path) end private def self.ensure_init unless @initialized fail ConfigInitiRequired, "ensure_init: Conf.init(app_root) should be invoked beforehand" end end end end