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 # # Extended variant of the #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 # @param config [Hash] a hash to be merged into the result, usually only recursivly called by the method itself # # @return [Hash] the resulting yaml config # def self.ext_load_file(yaml_path, inheritance_key='extends', extend_existing_arrays=true, config = {}) total_config ||= {} 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 merged_config = config.clone.deeper_merge(super_config, extend_existing_arrays: extend_existing_arrays) 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 super_config_path, inheritance_key, extend_existing_arrays, total_config.deeper_merge(merged_config, extend_existing_arrays: extend_existing_arrays) end total_config else delete_yaml_key inheritance_key, merged_config merged_config end end private # 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) base_path = File.dirname(caller_locations[1].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