require 'singleton'
require 'yaml'
require 'erb'
require 'hashie'
class Setting
class NotFound < RuntimeError; end
class FileError < RuntimeError; end
class AlreadyLoaded < RuntimeError; end
include Singleton
attr_reader :available_settings
# This method can be called only once.
#
# Parameter hash looks like this:
#
# { :files => [ "file1.yml", "file2.yml", ...],
# :path => "/var/www/apps/my-app/current/config/settings",
# :local => true }
#
# If :local => true is set, we will load all *.yml files under :path/local directory
# after all files in :files have been loaded. "Local" settings thus take precedence
# by design. See README for more details.
#
def self.load(args = {})
raise AlreadyLoaded.new('Settings already loaded') if self.instance.loaded?
self.instance.load(args)
end
def self.reload(args = {})
self.instance.load(args)
end
# In Method invocation syntax we collapse Hash values
# and return a single value if 'default' is found among keys
# or Hash has only one key/value pair.
#
# For example, if the YML data is:
# tax:
# default: 0.0
# california: 7.5
#
# Then calling Setting.tax returns "0.0""
#
# This is the preferred method of using settings class.
#
def self.method_missing(method, *args, &block)
self.instance.value_for(method, args) do |v, args|
self.instance.collapse_hashes(v, args)
end
end
def self.respond_to?(method_name, include_private = false)
self.instance.available_settings.has_key?(method_name.to_s.sub(/\?\z/, '')) ||
super
end
# In [] invocation syntax, we return settings value 'as is' without
# Hash conversions.
#
# For example, if the YML data is:
# tax:
# default: 0.0
# california: 7.5
#
# Then calling Setting['tax'] returns
# { 'default' => "0.0", 'california' => "7.5"}
def self.[](value)
self.instance.value_for(value)
end
# DEPRECATED: Please use method accessors instead.
def self.available_settings
self.instance.available_settings
end
#=================================================================
# Instance Methods
#=================================================================
def initialize
@available_settings ||= Hashie::Mash.quiet(:default).new
end
def has_key?(key)
@available_settings.has_key?(key) ||
(key[-1,1] == '?' && @available_settings.has_key?(key.chop))
end
def value_for(key, args = [])
name = key.to_s
raise NotFound.new("#{name} was not found") unless has_key?(name)
bool = false
if name[-1,1] == '?'
name.chop!
bool = true
end
v = @available_settings[name]
if block_given?
v = yield(v, args)
end
if v.is_a?(Integer) && bool
v.to_i > 0
else
v
end
end
# This method performs collapsing of the Hash settings values if the Hash
# contains 'default' value, or just 1 element.
def collapse_hashes(v, args)
out = if v.is_a?(Hash)
if args.empty?
if v.has_key?("default")
v['default'].nil? ? "" : v['default']
elsif v.keys.size == 1
v.values.first
else
v
end
else
v[args.shift.to_s]
end
else
v
end
if out.is_a?(Hash) && !args.empty?
collapse_hashes(out, args)
elsif out.is_a?(Hash) && out.has_key?('default')
out['default']
else
out
end
end
def loaded?
@loaded
end
def load(params)
# reset settings hash
@available_settings = Hashie::Mash.quiet(:default).new
@loaded = false
files = []
path = params[:path] || Dir.pwd
params[:files].each do |file|
files << File.join(path, file)
end
if params[:local]
files << Dir.glob(File.join(path, 'local', '*.yml')).sort
end
files.flatten.each do |file|
begin
# Ruby versions before 3.0.3 include Psych < 3.3.2, which does not include `unsafe_load`. In those versions,
# `load` is the behavior we want (in later versions, `load` uses `safe_load`, which doesn't support aliases and
# requires allowlisting classes used in files.
if Psych::VERSION < '3.3.2'
@available_settings.deep_merge!(YAML::load(ERB.new(IO.read(file)).result) || {}) if File.exist?(file)
else
@available_settings.deep_merge!(YAML::unsafe_load(ERB.new(IO.read(file)).result) || {}) if File.exist?(file)
end
rescue Exception => e
raise FileError.new("Error parsing file #{file}, with: #{e.message}")
end
end
@loaded = true
@available_settings
end
end