# # # @author Hollin Wilkins & Curtis Schofield & Mason Bo-bay-son require 'find' require 'digest/md5' module RubyConf def self.puts(logger = $stdout, *obj) logger.puts(*obj) if logger.respond_to?(:puts) end def self.err(*objs) self.puts($stderr, *objs) end def self.out(*objs) self.puts($stdout, *objs) end class Loader < BasicObject EXTENTIONS = %w{rbc rbcnf rbconf rbconfig rubyconf rubyconfig ruby-conf ruby-config} @@conf = @@path = @@mtime = @@md5 = nil class << self def __rc_loaded_conf() { path:@@path, mtime:@@mtime, md5:@@md5 } end def __rc_conf() @@conf end def __rc_set_conf(conf = nil) @@conf, @@path, @@mtime, @@md5 = conf, nil, nil, nil end def __rc_load(path) __rc_set_conf if load(path) && @@conf @@path, @@mtime, @@md5 = path, File.mtime(path).to_i, Digest::MD5.hexdigest(File.read(path)) RubyConf.out "[ruby-conf] Auto-Loaded config at path: #{path}" end end def __rc_reload if @@mtime && @@mtime != File.mtime(@@path).to_i && @@md5 != Digest::MD5.hexdigest(File.read(@@path)) RubyConf.err "[ruby-conf] Detected change in config file, reloading..." __rc_load(@@path) end end def method_missing(name, *args, &block) __rc_reload if @@conf.nil? Find.find('.') do |path| next unless @@conf.nil? && path =~ /\.(?:rb|config|conf|#{EXTENTIONS.join('|')})$/ if path =~ /\.(?:#{EXTENTIONS.join('|')})$/ || File.read(path) =~ /^\s*\#\s*\:\s*ruby-conf\s*$/mi break if __rc_load(path) end end end @@conf.__send__(name, *args, &block) end def to_s() __rc_reload @@conf.to_s end def to_str() __rc_reload @@conf.to_str end def inspect() __rc_reload @@conf.inspect end def nil?() __rc_reload @@conf.nil? end end end module Magic attr_accessor :__rc_chain def __rc_gather() "#{to_s}#{__rc_chain.nil? ? "" : " #{__rc_chain.__rc_gather}"}" end end class Config attr_reader :__rc_attributes, :__rc_parent, :__rc_name, :__rc_chains, :__rc_locked def __rc_root() __rc_parent ? __rc_parent.__rc_root : self end def detach(parent = nil) @__rc_parent = parent @__rc_attributes.values.each do |child| child.detach(self) if child.is_a?(Config) end self end def initialize(name = nil, parent = nil, &block) @__rc_locked, @__rc_attributes, @__rc_chains, @__rc_parent = false, {}, [], parent @__rc_name = name.to_sym if name instance_eval(&block) if block_given? __rc_lock end def __rc_lock @__rc_locked = true @__rc_attributes.each_value { |value| value.__rc_lock if value.is_a?(Config) } self end def [](name, *args) name = name.to_sym value = @__rc_attributes[name] if value.is_a?(Proc) args += [nil] * (value.arity.abs - args.size) if value.arity.abs > args.size @@stack ||= [] if @@stack.include?(name) RubyConf.err("[ruby-conf] Detected recursive proc: #{name}") "[RECURSIVE]" else @@stack << name begin __rc_root.instance_exec(*args, &value) ensure @@stack.delete(name) end end else value end end def __rc_copy(o) if o.is_a?(Hash) o.inject({}) { |copy, (key, val)| copy[key] = __rc_copy(val); copy } elsif o.is_a?(Array) o.inject([]) { |copy, i| copy << __rc_copy(i); copy } else o end end def to_a() [self] end def []=(name, value) @__rc_attributes[name.to_sym] = value end def __rc_set_defaults(conf, key, value) if conf.__rc_attributes.key?(key) if value.is_a?(Config) sub = conf.__rc_attributes[key] value.__rc_attributes.each do |k, v| __rc_set_defaults(sub, k, v) end end else conf.__rc_attributes[key] = value end end def method_missing(method_name, *args, &block) name, modifier = method_name.to_s.match(/^(?.*?)(?[?!])?$/).captures name = name.to_sym options = args.last.is_a?(Hash) ? args.last : {} if block_given? if options.key?(:inherits) self[name] = [*options[:inherits]].inject(Config.new(name, self, &block)) do |conf, inherited| inherited = self[inherited.to_sym] unless inherited.is_a?(Config) __rc_copy(inherited.__rc_attributes).each do |key, value| __rc_set_defaults(conf, key, value) end conf end elsif self[name].is_a?(Config) self[name].instance_eval(&block) else self[name] = Config.new(name, self, &block) end while (chain = self[name].__rc_chains.pop) self[name][chain] = chain.__rc_chain.nil? ? "" : chain.__rc_chain.__rc_gather end else if @__rc_locked && args.size == 0 && !@__rc_attributes.key?(name) return case modifier when '!' super when '?' false else nil end end if !@__rc_attributes.key?(name) && (args.size == 0 || args.first.is_a?(Magic)) str = name.to_s str.extend(Magic) if args.first.is_a?(Magic) str.__rc_chain = args.first @__rc_chains.delete_if { |a| a.equal? args.first } end @__rc_chains << str return str end if args.empty? modifier == '?' ? !!self[name] : self[name] else args = args.size == 1 ? args.first : args (@__rc_locked && __rc_attributes[name.to_sym].is_a?(Proc)) ? self[name, *args] : self[name] = args end end end def respond_to?(name) super || @__rc_attributes.key?(name) || @__rc_parent.respond_to?(name) end def __rc_build_string(depth = 0) str = "" str += "[#{@__rc_name || "CONFIG"}]\n" unless @__rc_parent str += "\n" @__rc_attributes.keys.map{|k| k.to_s }.sort.each do |key| value = begin self[key] rescue => e "[UNRESOLVED]" end str += " " * depth str += "#{key}:" str += value.is_a?(Config) ? (value == self ? " [SELF]\n" : value.__rc_build_string(depth+1)) : " #{value}\n" end str end def __rc_build_inspect() istr = "" istr += "[#{@__rc_name || "CONFIG"}] " unless @__rc_parent istr += @__rc_attributes.keys.map{|k| k.to_s }.sort.map{ |key| str = "" value = begin self[key] rescue => e "[UNRESOLVED:#{e}]" end str += "#{key}: " str += value.is_a?(RubyConf::Config) ? (value == self ? "[SELF]" : "{ #{value.__rc_build_inspect} }") : value.inspect str }.join(", ") istr end def to_s() __rc_build_string end def to_str() to_s end def inspect() __rc_build_inspect end end @@__rc_configs = {} class << self def clear() Loader::__rc_set_conf @@__rc_configs.clear end def to_s() @@__rc_configs.to_s end def to_str() @@__rc_configs.to_str end def inspect() @@__rc_configs.inspect end def [](name) @@__rc_configs[name.to_sym] end def method_missing(name, *args) @@__rc_configs[name.to_sym] end def respond_to?(name) @@__rc_configs.key?(name.to_sym) end def define(name = nil, options = {}, &block) config = Config.new(name, &block) @@__rc_configs[name.to_sym] = config unless name.nil? const = options.fetch(:as, name) if const && const.to_s[/^[A-Z]/] const = const.to_sym ::Object.const_set(const, config) if !::Object.const_defined?(const) || ::Object.const_get(const).is_a?(Config) end if Loader::__rc_conf.nil? && (name.nil? || name.to_s =~ /^(?:Rails)?Conf/) default_conf = if ::Object.const_defined?(:Rails) cfg = config[:"#{::Rails.env}"] || config[:"#{::Rails.env}_conf"] || config[:"#{::Rails.env}_config"] (cfg && cfg.detach) || config else config end Loader::__rc_set_conf(default_conf) end config end end end RUBY_CONF = RubyConf::Loader