module ROXML
unless const_defined? 'XML_PARSER'
begin
require 'libxml'
XML_PARSER = 'libxml' # :nodoc:
rescue LoadError
XML_PARSER = 'rexml' # :nodoc:
end
end
require File.join(File.dirname(__FILE__), 'xml', XML_PARSER)
class RequiredElementMissing < Exception # :nodoc:
end
#
# Internal base class that represents an XML - Class binding.
#
class XMLRef # :nodoc:
attr_reader :accessor
delegate :name, :required?, :array?, :default, :wrapper, :blocks, :to => :opts
alias_method :xpath_name, :name
def initialize(accessor, args)
@accessor = accessor
@opts = args
end
# Reads data from the XML element and populates the instance
# accordingly.
def populate(xml, instance)
data = value(xml)
instance.instance_variable_set("@#{accessor}", data) if data
instance
end
def name?
false
end
def update_xml(xml, value)
returning wrap(xml) do |xml|
write_xml(xml, value)
end
end
def value(xml)
value = fetch_value(xml)
if value.blank?
raise RequiredElementMissing if required?
value = default
end
apply_blocks(value)
end
private
attr_reader :opts
def apply_blocks(val)
blocks.each {|block| val = block[*val] } unless blocks.empty?
val
end
def xpath
wrapper ? "#{wrapper}/#{xpath_name}" : xpath_name.to_s
end
def wrap(xml)
(wrapper && xml.name != wrapper) ? xml.child_add(XML::Node.new_element(wrapper)) : xml
end
end
# Interal class representing an XML attribute binding
#
# In context:
#
# XMLTextRef
#
class XMLAttributeRef < XMLRef # :nodoc:
private
# Updates the attribute in the given XML block to
# the value provided.
def write_xml(xml, value)
xml.attributes[name] = value.to_s.to_utf
end
def fetch_value(xml)
attr = xml.search(xpath).first
attr && attr.value
end
def xpath_name
"@#{name}"
end
end
# Interal class representing XML content text binding
#
# In context:
#
# XMLTextRef
#
class XMLTextRef < XMLRef # :nodoc:
delegate :cdata?, :content?, :to => :opts
def name?
name == '*'
end
private
# Updates the text in the given _xml_ block to
# the _value_ provided.
def write_xml(xml, value)
if content?
add(xml, value)
elsif name?
xml.name = value
elsif array?
value.each do |v|
add(xml.child_add(XML::Node.new_element(name)), v)
end
else
add(xml.child_add(XML::Node.new_element(name)), value)
end
end
def fetch_value(xml)
if content?
xml.content.strip
elsif name?
xml.name
elsif array?
xml.search(xpath).collect do |e|
e.content.strip.to_latin if e.content
end
else
child = xml.search(name).first
child.content if child
end
end
def add(dest, value)
if cdata?
dest.child_add(XML::Node.new_cdata(value.to_s.to_utf))
else
dest.content = value.to_s.to_utf
end
end
end
class XMLHashRef < XMLTextRef # :nodoc:
delegate :hash, :to => :opts
private
# Updates the composed XML object in the given XML block to
# the value provided.
def write_xml(xml, value)
value.each_pair do |k, v|
node = xml.child_add(XML::Node.new_element(hash.wrapper))
hash.key.update_xml(node, k)
hash.value.update_xml(node, v)
end
end
def fetch_value(xml)
xml.search(xpath).collect do |e|
[hash.key.value(e), hash.value.value(e)]
end
end
def apply_blocks(vals)
unless blocks.empty?
vals.collect! do |kvp|
super(kvp)
end
end
vals.to_hash
end
end
class XMLObjectRef < XMLTextRef # :nodoc:
delegate :type, :to => :opts
private
# Updates the composed XML object in the given XML block to
# the value provided.
def write_xml(xml, value)
unless array?
xml.child_add(value.to_xml(name))
else
value.each do |v|
xml.child_add(v.to_xml(name))
end
end
end
def fetch_value(xml)
unless array?
if child = xml.search(xpath).first
instantiate(child)
end
else
xml.search(xpath).collect do |e|
instantiate(e)
end
end
end
def instantiate(elem)
if type.respond_to? :from_xml
type.from_xml(elem)
else
type.new(elem)
end
end
end
end