require 'closed_struct' require 'yaml' require 'erb' class ApplicationConfiguration # Create a new ApplicationConfiguration object. conf_path_1 is the path to your YAML configuration file. # If conf_path_2 is given, the contents are recursively merged with the contents of conf_path_1. # This allows you to have a "base" configuration with settings that are overrided by "environment specific" # (developement, test, production, etc) settings. # # Ex: # ApplicationConfiguration.new(RAILS_ROOT+"/config/base.yml", RAILS_ROOT+"/environments/#{RAILS_ENV}_config.yml") def initialize(conf_path_1, conf_path_2 = nil) @conf_path_1, @conf_path_2 = conf_path_1, conf_path_2 reload! end # Rereads your configuration files and rebuilds your ApplicationConfiguration object. This is useful # for when you edit your config files, but don't want to restart your web server. def reload! conf1 = load_conf_file(@conf_path_1) conf2 = load_conf_file(@conf_path_2) @config_hash = recursive_merge(conf1, conf2) @config = ClosedStruct.r_new(@config_hash) end def use_environment!(environment, options = {}) raise ArgumentError, "environment doesn't exist in app config: #{environment}" \ unless @config_hash.has_key?(environment.to_s) @config_hash = @config_hash[environment.to_s] @config = @config.send(environment) if options[:override_with] and File.exist?(options[:override_with]) overriding_config = load_conf_file(options[:override_with]) @config_hash = recursive_merge(@config_hash, overriding_config) @config = ClosedStruct.r_new(@config_hash) end end private def method_missing(name, *args) if @config.respond_to?(name) @config.send(name, *args) else super end end def load_conf_file(conf_path) return {} if !conf_path or conf_path.empty? File.open(conf_path, "r") do |file| YAML.load(ERB.new(file.read).result) || {} end end # Recursively merges hashes. h2 will overwrite h1. def recursive_merge(h1, h2) #:nodoc: h1.merge(h2){ |k, v1, v2| v2.kind_of?(Hash) ? recursive_merge(v1, v2) : v2 } end end