Sha256: a956dbe190a661e1e314112699bf8307a78b80780c9d5cc84e394367d494cf1c

Contents?: true

Size: 1.98 KB

Versions: 1

Compression:

Stored size: 1.98 KB

Contents

require "libxml"
# Portions of this code were copied and modified from Ruby on Rails, released
# under the MIT license, copyright (c) 2005-2009 David Heinemeier Hansson
module Braintree
  module Xml
    module Libxml # :nodoc:
      LIB_XML_LIMIT = 30000000

      def self.parse(xml_string)
        old_keep_blanks_setting = ::LibXML::XML.default_keep_blanks
        ::LibXML::XML.default_keep_blanks = false
        root_node = LibXML::XML::Parser.string(xml_string.strip).parse.root
        _node_to_hash(root_node)
      ensure
        ::LibXML::XML.default_keep_blanks = old_keep_blanks_setting
      end

      def self._node_to_hash(node, hash = {})
        if node.text?
          raise ::LibXML::XML::Error if node.content.length >= LIB_XML_LIMIT
          hash[CONTENT_ROOT] = node.content
        else
          sub_hash = _build_sub_hash(hash, node.name)
          _attributes_to_hash(node, sub_hash)
          if _array?(node)
            _children_array_to_hash(node, sub_hash)
          else
            _children_to_hash(node, sub_hash)
          end
        end
        hash
      end

      def self._build_sub_hash(hash, name)
        sub_hash = {}
        if hash[name]
          if !hash[name].kind_of? Array
            hash[name] = [hash[name]]
          end
          hash[name] << sub_hash
        else
          hash[name] = sub_hash
        end
        sub_hash
      end

      def self._children_to_hash(node, hash={})
        node.each { |child| _node_to_hash(child, hash) }
        _attributes_to_hash(node, hash)
        hash
      end

      def self._attributes_to_hash(node, hash={})
        node.each_attr { |attr| hash[attr.name] = attr.value }
        hash
      end

      def self._children_array_to_hash(node, hash={})
        hash[node.child.name] = node.map do |child|
          _children_to_hash(child, {})
        end
        hash
      end

      def self._array?(node)
        node.child? && node.child.next? && node.child.name == node.child.next.name
      end
    end
  end
end

Version data entries

1 entries across 1 versions & 1 rubygems

Version Path
braintree-3.4.0 lib/braintree/xml/libxml.rb