module ROXML class RequiredElementMissing < Exception # :nodoc: end # # Internal base class that represents an XML - Class binding. # class XMLRef # :nodoc: delegate :required?, :array?, :blocks, :accessor, :default, :to => :opts def initialize(opts, instance) @opts = opts @instance = instance end def to_xml val = @instance.__send__(accessor) opts.to_xml.respond_to?(:call) ? opts.to_xml.call(val) : val end def update_xml(xml, value) returning wrap(xml) do |xml| write_xml(xml, value) end end def name opts.name_explicit? ? opts.name : conventionize(opts.name) end def xpath_name if !opts.name_explicit? && namespace = @instance.class.roxml_namespace "#{namespace}:#{name}" else name end end def value_in(xml) xml = XML::Node.from(xml) value = fetch_value(xml) value = default if value.nil? freeze(apply_blocks(value)) end private attr_reader :opts def conventionize(what) convention ||= @instance.class.respond_to?(:roxml_naming_convention) && @instance.class.roxml_naming_convention if !what.blank? && convention.respond_to?(:call) URI.unescape(convention.call(URI.escape(what, /\/|::/))) else what end end def wrapper conventionize(opts.wrapper) end def apply_blocks(val) begin blocks.apply_to(val) rescue Exception => ex raise ex, "#{accessor}: #{ex.message}" end end def freeze(val) if opts.frozen? val.each(&:freeze) if val.is_a?(Enumerable) val.freeze else val end end def xpath wrapper ? "#{wrapper}/#{xpath_name}" : xpath_name.to_s end def auto_xpath "#{conventionize(opts.name.pluralize)}/#{xpath_name}" if array? end def wrap(xml) return xml if !wrapper || xml.name == wrapper if child = xml.children.find {|c| c.name == wrapper } return child end xml.add_child(XML::Node.new(wrapper.to_s)) end def nodes_in(xml) vals = xml.search(xpath) if (opts.hash? || opts.array?) && vals.empty? && !wrapper && auto_xpath vals = xml.search(auto_xpath) @auto_vals = !vals.empty? end if vals.empty? raise RequiredElementMissing, "#{name} from #{xml} for #{accessor}" if required? default else yield(vals) end 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 end def fetch_value(xml) nodes_in(xml) do |nodes| nodes.first.value end end def xpath_name "@#{name}" end end # Interal class representing XML content text binding # # In context: # # XMLTextRef # class XMLTextRef < XMLRef # :nodoc: delegate :cdata?, :content?, :name?, :to => :opts 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.add_child(XML::Node.new(name)), v) end else add(xml.add_child(XML::Node.new(name)), value) end end def fetch_value(xml) if content? || name? value = if content? xml.content.to_s.strip elsif name? xml.name end if value.empty? raise RequiredElementMissing, "#{name} from #{xml} for #{accessor}" if required? default else value end else nodes_in(xml) do |nodes| if array? nodes.collect do |e| e.content.strip end else nodes.first.content end end end end def add(dest, value) if cdata? dest.child_add(XML::Node.new_cdata(value.to_s)) else dest.content = value.to_s end end end class XMLHashRef < XMLTextRef # :nodoc: delegate :hash, :to => :opts def initialize(opts, inst) super(opts, inst) @key = opts.hash.key.to_ref(inst) @value = opts.hash.value.to_ref(inst) end 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(hash.wrapper)) @key.update_xml(node, k) @value.update_xml(node, v) end end def fetch_value(xml) nodes_in(xml) do |nodes| nodes.collect do |e| [@key.value_in(e), @value.value_in(e)] end end end def apply_blocks(vals) unless blocks.empty? vals.collect! do |kvp| super(kvp) end end to_hash(vals) if vals end def freeze(vals) if opts.frozen? vals.each_pair{|k, v| k.freeze; v.freeze } vals.freeze else vals end end def to_hash(array) hash = array.inject({}) do |result, (k, v)| result[k] ||= [] result[k] << v result end hash.each_pair do |k, v| hash[k] = v.first if v.one? end 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) if array? value.each do |v| xml.child_add(v.to_xml(name)) end elsif value.is_a?(ROXML) xml.child_add(value.to_xml(name)) else node = XML::Node.new(name) node.content = value.to_xml xml.child_add(node) end end def fetch_value(xml) nodes_in(xml) do |nodes| unless array? instantiate(nodes.first) else nodes.collect do |e| instantiate(e) end end end end def instantiate(elem) if type.respond_to? :from_xml type.from_xml(elem) else type.new(elem) end end end end