lib/safe_yaml/resolver.rb



module SafeYAML
  class Resolver
    def initialize(options)
      @options              = SafeYAML::OPTIONS.merge(options || {})
      @whitelist            = @options[:whitelisted_tags] || []
      @initializers         = @options[:custom_initializers] || {}
      @raise_on_unknown_tag = @options[:raise_on_unknown_tag]
    end

    def resolve_node(node)
      return node if !node
      return self.native_resolve(node) if tag_is_whitelisted?(self.get_node_tag(node))

      case self.get_node_type(node)
      when :root
        resolve_root(node)
      when :map
        resolve_map(node)
      when :seq
        resolve_seq(node)
      when :scalar
        resolve_scalar(node)
      when :alias
        resolve_alias(node)
      else
        raise "Don't know how to resolve this node: #{node.inspect}"
      end
    end

    def resolve_map(node)
      tag  = get_and_check_node_tag(node)
      hash = @initializers.include?(tag) ? @initializers[tag].call : {}
      map  = normalize_map(self.get_node_value(node))

      # Take the "<<" key nodes first, as these are meant to approximate a form of inheritance.
      inheritors = map.select { |key_node, value_node| resolve_node(key_node) == "<<" }
      inheritors.each do |key_node, value_node|
        merge_into_hash(hash, resolve_node(value_node))
      end

      # All that's left should be normal (non-"<<") nodes.
      (map - inheritors).each do |key_node, value_node|
        hash[resolve_node(key_node)] = resolve_node(value_node)
      end

      return hash
    end

    def resolve_seq(node)
      seq = self.get_node_value(node)

      tag = get_and_check_node_tag(node)
      arr = @initializers.include?(tag) ? @initializers[tag].call : []

      seq.inject(arr) { |array, n| array << resolve_node(n) }
    end

    def resolve_scalar(node)
      Transform.to_proper_type(self.get_node_value(node), self.value_is_quoted?(node), get_and_check_node_tag(node), @options)
    end

    def get_and_check_node_tag(node)
      tag = self.get_node_tag(node)
      SafeYAML.tag_safety_check!(tag, @options)
      tag
    end

    def tag_is_whitelisted?(tag)
      @whitelist.include?(tag)
    end

    def options
      @options
    end

    private
    def normalize_map(map)
      # Syck creates Hashes from maps.
      if map.is_a?(Hash)
        map.inject([]) { |arr, key_and_value| arr << key_and_value }

      # Psych is really weird; it flattens out a Hash completely into: [key, value, key, value, ...]
      else
        map.each_slice(2).to_a
      end
    end

    def merge_into_hash(hash, array)
      array.each do |key, value|
        hash[key] = value
      end
    end
  end
end