lib/yaml_extend.rb in yaml_extend-1.0.0 vs lib/yaml_extend.rb in yaml_extend-1.0.1
- old
+ new
@@ -1,137 +1,143 @@
-require 'yaml_extend/version'
-
-require 'yaml'
-require 'deep_merge/rails_compat'
-
-require_relative 'custom_errors/invalid_key_type_error'
-
-#
-# Extending the YAML library to allow to inherit from another YAML file(s)
-#
-
-module YAML
- # default path in the yaml file where the files to inherit from are defined
- DEFAULT_INHERITANCE_KEY = 'extends'
- @@ext_load_key = nil
- #
- # Set a custom inheritance key globally once.
- # So you don't need to specify it on every call of ext_load_file()
- #
- # @param key [String|Array<String>|nil] the key in the yaml file, containing the file strings to extend from. Set nil or call #reset_load_key to reset the key.
- def self.ext_load_key=(key)
- if key.is_a?(String) || key.is_a?(Array) || key.nil?
- @@ext_load_key = key
- else
- raise "Parameter 'key' must be of type String or Array<String> or nil"
- end
- end
-
- #
- # Reset the ext_load_key and use the default settings
- #
- def self.reset_load_key()
- @@ext_load_key = nil
- end
-
- #
- # Extended variant of the YAML.load_file method by providing the
- # ability to inherit from other YAML file(s)
- #
- # @param yaml_path [String] the path to the yaml file to be loaded
- # @param inheritance_key [String|Array] The key used in the yaml file to extend from another YAML file. Use an Array if you want to use a tree structure key like "options.extends" => ['options','extends']
- # @param extend_existing_arrays [Boolean] extend existing arrays instead of replacing them
- # @return [Hash] the resulting yaml config
- #
- def self.ext_load_file(yaml_path, inheritance_key=nil, extend_existing_arrays=true)
- YAML.ext_load_file_recursive(yaml_path, inheritance_key, extend_existing_arrays, {})
- end
-
- private
-
- #
- # @param config [Hash] a hash to be merged into the result, usually only recursivly called by the method itself
- #
- def self.ext_load_file_recursive(yaml_path, inheritance_key, extend_existing_arrays, config)
- private_class_method
- if inheritance_key.nil?
- inheritance_key = @@ext_load_key || DEFAULT_INHERITANCE_KEY
- end
- total_config = config.clone
-
- yaml_path = YAML.make_absolute_path yaml_path
- super_config = YAML.load_file(File.open(yaml_path))
- super_inheritance_files = yaml_value_by_key inheritance_key, super_config
- delete_yaml_key inheritance_key, super_config # we don't merge the super inheritance keys into the base yaml
-
- if super_inheritance_files && super_inheritance_files != ''
- super_inheritance_files = [super_inheritance_files] unless super_inheritance_files.is_a? Array # we support strings as well as arrays of type string to extend from
- super_inheritance_files.each_with_index do |super_inheritance_file, index|
- super_config_path = File.dirname(yaml_path) + '/' + super_inheritance_file
- total_config = YAML.ext_load_file_recursive(super_config_path, inheritance_key, extend_existing_arrays, total_config)
- end
- end
- total_config.deeper_merge!(super_config, extend_existing_arrays: extend_existing_arrays)
- end
-
- # some logic to ensure absolute file inheritance as well as
- # relative file inheritance in yaml files
- def self.make_absolute_path(file_path)
- private_class_method
- return file_path if YAML.absolute_path?(file_path) && File.exist?(file_path)
- # caller_locations returns the current execution stack
- # [0] is the call from ext_load_file_recursive,
- # [1] is inside ext_load_file,
- # [2] is the exteranl caller of YAML.ext_load_file
- base_path = File.dirname(caller_locations[2].path)
- return base_path + '/' + file_path if File.exist? base_path + '/' + file_path # relative path from yaml file
- return Dir.pwd + '/' + file_path if File.exist? Dir.pwd + '/' + file_path # relative path from project
- error_message = "Can not find absolute path of '#{file_path}'"
- unless YAML.absolute_path? file_path
- error_message += "\nAlso tried:\n- #{base_path + '/' + file_path}\n"\
- "- #{Dir.pwd + '/' + file_path}\n"
- end
- raise error_message
- end
-
- def self.absolute_path?(path)
- private_class_method
- path.start_with?('/') || # unix like
- (path.length >= 3 && path[1] == ':') # ms windows
- end
-
- # Return the value of the corresponding key
- # @param key [String|Array]
- def self.yaml_value_by_key(key, config)
- return config[key] if key.is_a? String
- if valid_key_type? key
- cfg_copy = config.clone
- key.each do |key|
- if cfg_copy.nil?
- return
- elsif valid_key_type? key
- cfg_copy = cfg_copy[key]
- end
- end
- cfg_copy
- end
- end
-
- def self.valid_key_type?(key)
- key.is_a?(Array) || key.is_a?(String) ||
- raise(InvalidKeyTypeError,"Invalid key of type '#{key.class.name}'. Valid types are String and Array.")
- end
-
- def self.delete_yaml_key(key, config)
- return config.delete(key) if key.is_a? String
- cfg_ref = config
- last_ref = nil
- key.each do |key|
- if valid_key_type?(key) && !cfg_ref[key].nil?
- last_ref = cfg_ref
- cfg_ref = cfg_ref[key] unless cfg_ref.nil?
- end
- end
- last_ref.delete key.last unless last_ref.nil?
- end
-
-end
+require 'yaml_extend/version'
+
+require 'yaml'
+require 'deep_merge/rails_compat'
+
+require_relative 'custom_errors/invalid_key_type_error'
+
+#
+# Extending the YAML library to allow to inherit from another YAML file(s)
+#
+
+module YAML
+ # default path in the yaml file where the files to inherit from are defined
+ DEFAULT_INHERITANCE_KEY = 'extends'
+ @@ext_load_key = nil
+ #
+ # Set a custom inheritance key globally once.
+ # So you don't need to specify it on every call of ext_load_file()
+ #
+ # @param key [String|Array<String>|nil] the key in the yaml file, containing the file strings to extend from. Set nil or call #reset_load_key to reset the key.
+ def self.ext_load_key=(key)
+ if key.is_a?(String) || key.is_a?(Array) || key.nil?
+ @@ext_load_key = key
+ else
+ raise "Parameter 'key' must be of type String or Array<String> or nil"
+ end
+ end
+
+ #
+ # Reset the ext_load_key and use the default settings
+ #
+ def self.reset_load_key()
+ @@ext_load_key = nil
+ end
+
+ #
+ # Extended variant of the YAML.load_file method by providing the
+ # ability to inherit from other YAML file(s)
+ #
+ # @param yaml_path [String] the path to the yaml file to be loaded
+ # @param inheritance_key [String|Array] The key used in the yaml file to extend from another YAML file. Use an Array if you want to use a tree structure key like "options.extends" => ['options','extends']
+ # @param extend_existing_arrays [Boolean] extend existing arrays instead of replacing them
+ # @return [Hash] the resulting yaml config
+ #
+ def self.ext_load_file(yaml_path, inheritance_key=nil, extend_existing_arrays=true)
+ YAML.ext_load_file_recursive(yaml_path, inheritance_key, extend_existing_arrays, {})
+ end
+
+ private
+
+ #
+ # @param config [Hash] a hash to be merged into the result, usually only recursivly called by the method itself
+ #
+ def self.ext_load_file_recursive(yaml_path, inheritance_key, extend_existing_arrays, config)
+ private_class_method
+ if inheritance_key.nil?
+ inheritance_key = @@ext_load_key || DEFAULT_INHERITANCE_KEY
+ end
+ total_config = config.clone
+
+ yaml_path = YAML.make_absolute_path yaml_path
+ super_config = YAML.load_file(File.open(yaml_path))
+ super_inheritance_files = yaml_value_by_key inheritance_key, super_config
+ delete_yaml_key inheritance_key, super_config # we don't merge the super inheritance keys into the base yaml
+
+ if super_inheritance_files && super_inheritance_files != ''
+ super_inheritance_files = [super_inheritance_files] unless super_inheritance_files.is_a? Array # we support strings as well as arrays of type string to extend from
+ super_inheritance_files.each_with_index do |super_inheritance_file, index|
+ # Extend a YAML path in an absolute directory
+ if YAML.absolute_path?(super_inheritance_file)
+ super_config_path = YAML.make_absolute_path(super_inheritance_file)
+ # Extend a YAML path in a relative directory
+ else
+ super_config_path = File.dirname(yaml_path) + '/' + super_inheritance_file
+ end
+ total_config = YAML.ext_load_file_recursive(super_config_path, inheritance_key, extend_existing_arrays, total_config)
+ end
+ end
+ total_config.deeper_merge!(super_config, extend_existing_arrays: extend_existing_arrays)
+ end
+
+ # some logic to ensure absolute file inheritance as well as
+ # relative file inheritance in yaml files
+ def self.make_absolute_path(file_path)
+ private_class_method
+ return file_path if YAML.absolute_path?(file_path) && File.exist?(file_path)
+ # caller_locations returns the current execution stack
+ # [0] is the call from ext_load_file_recursive,
+ # [1] is inside ext_load_file,
+ # [2] is the exteranl caller of YAML.ext_load_file
+ base_path = File.dirname(caller_locations[2].path)
+ return base_path + '/' + file_path if File.exist? base_path + '/' + file_path # relative path from yaml file
+ return Dir.pwd + '/' + file_path if File.exist? Dir.pwd + '/' + file_path # relative path from project
+ error_message = "Can not find absolute path of '#{file_path}'"
+ unless YAML.absolute_path? file_path
+ error_message += "\nAlso tried:\n- #{base_path + '/' + file_path}\n"\
+ "- #{Dir.pwd + '/' + file_path}\n"
+ end
+ raise error_message
+ end
+
+ def self.absolute_path?(path)
+ private_class_method
+ path.start_with?('/') || # unix like
+ (path.length >= 3 && path[1] == ':') # ms windows
+ end
+
+ # Return the value of the corresponding key
+ # @param key [String|Array]
+ def self.yaml_value_by_key(key, config)
+ return config[key] if key.is_a? String
+ if valid_key_type? key
+ cfg_copy = config.clone
+ key.each do |key|
+ if cfg_copy.nil?
+ return
+ elsif valid_key_type? key
+ cfg_copy = cfg_copy[key]
+ end
+ end
+ cfg_copy
+ end
+ end
+
+ def self.valid_key_type?(key)
+ key.is_a?(Array) || key.is_a?(String) ||
+ raise(InvalidKeyTypeError,"Invalid key of type '#{key.class.name}'. Valid types are String and Array.")
+ end
+
+ def self.delete_yaml_key(key, config)
+ return config.delete(key) if key.is_a? String
+ cfg_ref = config
+ last_ref = nil
+ key.each do |key|
+ if valid_key_type?(key) && !cfg_ref[key].nil?
+ last_ref = cfg_ref
+ cfg_ref = cfg_ref[key] unless cfg_ref.nil?
+ end
+ end
+ last_ref.delete key.last unless last_ref.nil?
+ end
+
+end