require "rubygems"
require "hpricot"
# Apricot eats Gorilla is a SOAP communication helper.
#
# It translates between SOAP response messages (XML) and Ruby Hashes and may
# be used to build a SOAP request envelope. It is based on CobraVsMongoose but
# uses Hpricot instead of REXML and doesn't follow the BadgerFish convention.
#
# === xml_to_hash(xml, root_node = nil)
#
# xml = '
#
#
#
#
# secret
# example
#
#
#
#
# '
#
# ApricotEatsGorilla[xml, "//return"]
# # => { :auth_value => { :token => "secret", :client => "example" } }
#
# === hash_to_xml(hash)
#
# hash = { :apricot => { :eats => "Gorilla" } }
#
# ApricotEatsGorilla[hash]
# # => "Gorilla"
#
# === soap_envelope(namespaces = {})
#
# ApricotEatsGorilla.soap_envelope { "me" }
#
# # => '
# # =>
# # => me
# # =>
# # => '
class ApricotEatsGorilla
class << self # Class methods
# Flag to enable optional sorting of Hash keys.
attr_accessor :sort_keys
# Flag to disable conversion of tag names to lowerCamelCase.
attr_accessor :disable_tag_names_to_lower_camel_case
# Flag to disable conversion of Hash keys to snake_case.
attr_accessor :disable_hash_keys_to_snake_case
# Flag to disable conversion of Hash keys to Symbols.
attr_accessor :disable_hash_keys_to_symbol
# Array of XML nodes to add a namespace to.
attr_accessor :nodes_to_namespace
# The namespace for nodes in :nodes_to_namespace.
attr_accessor :node_namespace
# Shortcut method for translating between XML Strings and Ruby Hashes.
# Delegates to xml_to_hash in case +source+ is of type String or delegates
# to hash_to_xml in case +source+ is of type Hash. Returns nil otherwise.
#
# ==== Parameters
#
# * +source+ - The XML String or Ruby Hash to translate.
# * +root_node+ - Optional. Custom root node to start parsing the given XML.
def [](source, root_node = nil)
case source
when String
xml_to_hash(source, root_node)
when Hash
hash_to_xml(source)
else
nil
end
end
# Yields this class object in case a +block+ was given. Nice way for setting
# multiple options at once.
def setup
yield self if block_given?
end
# Converts a given +xml+ String into a Ruby Hash. Starts parsing at root
# node by default. The optional +root_node+ parameter can be used to specify
# a custom root node to start parsing at via XPath (Hpricot search).
# The root node itself won't be included in the Hash.
#
# ==== Parameters
#
# * +xml+ - The XML String to translate into a Ruby Hash.
# * +root_node+ - Optional. Custom root node to start parsing the given XML.
#
# ==== Examples
#
# xml = "Gorilla"
# ApricotEatsGorilla[xml]
# # => { :eats => "Gorilla" }
#
# xml = "Gorillas"
# ApricotEatsGorilla[xml, "//eats"]
# # => { :lots_of => "Gorillas" }
def xml_to_hash(xml, root_node = nil)
xml = clean_xml(xml)
doc = Hpricot.XML(xml)
root = root_node ? doc.at(root_node) : doc.root
return nil if root.nil?
xml_node_to_hash(root)
end
# Converts a given +hash+ into an XML String.
#
# ==== Parameters
#
# * +hash+ - The Ruby Hash to translate into an XML String.
#
# ==== Examples
#
# hash = { :apricot => { :eats => "Gorilla" } }
# ApricotEatsGorilla[hash]
# # => "Gorilla"
#
# hash = { :apricot => { :eats => [ "Gorillas", "Snakes" ] } }
# ApricotEatsGorilla[hash]
# # => "GorillasSnakes"
def hash_to_xml(hash)
nested_data_to_xml(hash.keys.first, hash.values.first)
end
# Builds a SOAP request envelope and includes the content from a given
# +block+ into the envelope body. Accepts a Hash of +namespaces+ and their
# corresponding URI for specifying custom namespaces.
#
# ==== Parameters
#
# * +namespaces+ - A Hash of namespaces and their corresponding URI.
#
# ==== Examples
#
# ApricotEatsGorilla.soap_envelope { "me" }
#
# # => '
# # =>
# # => me
# # =>
# # => '
#
# ApricotEatsGorilla.soap_envelope :wsdl => "http://example.com" do
# "me"
# end
#
# # => ' xmlns:wsdl="http://example.com"
# # => xmlns:env="http://schemas.xmlsoap.org/soap/envelope/">
# # =>
# # => me
# # =>
# # => '
def soap_envelope(namespaces = {})
namespaces["env"] = "http://schemas.xmlsoap.org/soap/envelope/"
tag("env:Envelope", namespaces) do
tag("env:Body") do
yield if block_given?
end
end
end
# Converts a given +string+ from CamelCase/lowerCamelCase to snake_case.
def to_snake_case(string)
string = string.gsub(/[A-Z]+/, '\1_\0').downcase
string = string[1, string.length-1] if string[0, 1] == "_"
string
end
# Converts a given +string+ from snake_case to lowerCamelCase.
def to_lower_camel_case(string)
string.to_s.gsub(/_(.)/) { $1.upcase }
end
private
# Actual implementation for xml_to_hash. Takes and iterates through a given
# Hpricot +element+ and returns a Ruby Hash equal to the given content.
#
# ==== Parameters
#
# * +element+ - The Hpricot element to translate into a Ruby Hash.
def xml_node_to_hash(element)
this_node = {}
element.each_child do |child|
# hpricot 0.6.1 returns an empty array, while 0.8 returns nil
if child.children.nil? || child.children.empty?
key, value = child.name, nil
elsif child.children.size == 1 && child.children.first.text?
key, value = child.name, booleanize(child.children.first.to_html)
else
key, value = child.name, xml_node_to_hash(child)
end
key = remove_namespace(key)
key = to_snake_case(key) unless disable_hash_keys_to_snake_case
key = key.intern unless disable_hash_keys_to_symbol
current = this_node[key]
case current
when Array
this_node[key] << value
when nil
this_node[key] = value
else
this_node[key] = [current.dup, value]
end
end
this_node
end
# Actual implementation for hash_to_xml. Takes a Hash key +name+ and a
# value +item+ and returns an XML String equal to the given content.
#
# ==== Parameters
#
# * +name+ - A Hash key to translate into an XML String.
# * +item+ - A Hash value to translate into an XML String.
def nested_data_to_xml(name, item)
case item
when Array
item.map { |subitem| nested_data_to_xml(name, subitem) }.join
when Hash
tag(name) do
opt_order(item).map { |tag, value|
case value
when Array
value.map { |subitem| nested_data_to_xml(tag, subitem) }.join
when Hash
nested_data_to_xml(tag, value)
else
tag(tag) { value.to_s } if value.respond_to? :to_s
end
}.join
end
else
tag(name) { item.to_s } if item.respond_to? :to_s
end
end
# Creates an XML tag. Expects a block for tag content. Defaults to an empty
# element tag in case no block was supplied.
#
# ==== Parameters
#
# * +name+ - The name of the XML tag.
# * +attributes+ - Optional. Hash of attributes for the XML tag.
def tag(name, attributes = {})
name = to_lower_camel_case(name) unless disable_tag_names_to_lower_camel_case
if nodes_to_namespace.kind_of? Array
name = "#{node_namespace}:#{name}" if node_namespace && nodes_to_namespace.include?(name)
end
return "<#{name} />" unless block_given?
attr = opt_order(attributes).map { |k, v| %Q( xmlns:#{k}="#{v}") }.to_s
body = (yield && !yield.empty?) ? yield : ""
"<#{name}#{attr}>" << body << "#{name}>"
end
# Removes line breaks and whitespace between tags from a given +xml+ String.
def clean_xml(xml)
xml.gsub!(/\n+/, "")
xml.gsub!(/(>)\s*(<)/, '\1\2')
xml
end
# Removes the namespace from a given XML +tag+.
def remove_namespace(tag)
tag.sub!(/.+:(.+)/, '\1')
tag
end
# Checks to see if a given +string+ matches "true" or "false" and converts
# these values to actual Boolean objects. Returns the original string in
# case it does not match "true" or "false".
def booleanize(string)
return true if string == "true"
return false if string == "false"
string
end
# Returns a sorted version of a given +hash+ if :sort_keys was enabled.
def opt_order(hash)
return hash unless sort_keys
hash.keys.sort_by { |s| s.to_s }.map { |key| [key, hash[key]] }
end
end
end