require 'rubygems'
require 'extensions/enumerable'
require 'extensions/array'
require 'extensions/object'
require 'activesupport'
%w(extensions/array extensions/string options xml).each do |file|
require File.join(File.dirname(__FILE__), 'roxml', file)
end
module ROXML # :nodoc:
def self.included(base) # :nodoc:
base.extend ClassMethods::Declarations
base.extend ClassMethods::Accessors
base.extend ClassMethods::Operations
base.class_eval do
include InstanceMethods::Accessors
include InstanceMethods::Conversions
end
end
module InstanceMethods # :nodoc:
# Instance method equivalents of the Class method accessors
module Accessors
# Provides access to ROXML::ClassMethods::Accessors::tag_name directly from an instance of a ROXML class
def tag_name
self.class.tag_name
end
# Provides access to ROXML::ClassMethods::Accessors::tag_refs directly from an instance of a ROXML class
def tag_refs
self.class.tag_refs
end
end
module Conversions
# Returns a LibXML::XML::Node or a REXML::Element representing this object
def to_xml(name = nil)
returning XML::Node.new_element(name || tag_name) do |root|
tag_refs.each do |ref|
if v = __send__(ref.accessor)
ref.update_xml(root, v)
end
end
end
end
end
end
# This class defines the annotation methods that are mixed into your
# Ruby classes for XML mapping information and behavior.
#
# See xml_name, xml_construct, xml, xml_reader and xml_accessor for
# available annotations.
#
module ClassMethods # :nodoc:
module Declarations
# Sets the name of the XML element that represents this class. Use this
# to override the default lowercase class name.
#
# Example:
# class BookWithPublisher
# xml_name :book
# end
#
# Without the xml_name annotation, the XML mapped tag would have been "bookwithpublisher".
#
def xml_name(name)
@tag_name = name
end
# Declares an accesser to a certain xml element, whether an attribute, a node,
# or a typed collection of nodes
#
# [sym] Symbol representing the name of the accessor
#
# == Type options
# All type arguments may be used as the type argument to indicate just type,
# or used as :from, pointing to a xml name to indicate both type and attribute name.
# Also, any type may be passed via an array to indicate that multiple instances
# of the object should be returned as an array.
#
# === :attr
# Declare an accessor that represents an XML attribute.
#
# Example:
# class Book
# xml_reader :isbn, :attr => "ISBN" # 'ISBN' is used to specify :from
# xml_accessor :title, :attr # :from defaults to :title
# end
#
# To map:
#
#
# === :text
# The default type, if none is specified. Declares an accessor that
# represents a text node from XML.
#
# Example:
# class Book
# xml :author, false, :text => 'Author'
# xml_accessor :description, :text, :as => :cdata
# xml_reader :title
# end
#
# To map:
#
# Programming Ruby: the pragmatic programmers' guide
#
# David Thomas
#
#
# Likewise, a number of :text node values can be collected in an array like so:
#
# Example:
# class Library
# xml_reader :books, [:text], :in => 'books'
# end
#
# To map:
#
#
# To kill a mockingbird
# House of Leaves
# Gödel, Escher, Bach
#
#
#
# === :content
# A special case of :text, this refers to the content of the current node,
# rather than a sub-node
#
# Example:
# class Contributor
# xml_reader :name, :content
# xml_reader :role, :attr
# end
#
# To map:
# James Wick
#
# === Hash
# Somewhere between the simplicity of a :text/:attr mapping, and the complexity of
# a full Object/Type mapping, lies the Hash mapping. It serves in the case where you have
# a collection of key-value pairs represented in your xml. You create a hash declaration by
# passing a hash mapping as the type argument. A few examples:
#
# ==== Hash of :attrs
# For xml such as this:
#
#
#
#
#
#
#
#
# You can use the :attrs key in you has with a [:key, :value] name array:
#
# xml_reader :definitions, {:attrs => ['dt', 'dd']}, :in => :definitions
#
# ==== Hash of :texts
# For xml such as this:
#
#
#
#
#
#
#
#
#
#
#
#
# You can individually declare your key and value names:
# xml_reader :definitions, {:key => 'word',
# :value => 'meaning'}
#
# ==== Hash of :content &c.
# For xml such as this:
#
#
# adjective: (of a geological formation) sloping downward from the center in all directions.
# To use evasions or ambiguities; equivocate.
#
#
# You can individually declare the key and value, but with the attr, you need to provide both the type
# and name of that type (i.e. {:attr => :word}), because omitting the type will result in ROXML
# defaulting to :text
# xml_reader :definitions, {:key => {:attr => 'word'},
# :value => :content}
#
# ==== Hash of :name &c.
# For xml such as this:
#
#
# adjective: (of a geological formation) sloping downward from the center in all directions.
# To use evasions or ambiguities; equivocate.
#
#
# You can pick up the node names (e.g. quaquaversally) using the :name keyword:
# xml_reader :definitions, {:key => :name,
# :value => :content}
#
# === Other ROXML Class
# Declares an accessor that represents another ROXML class as child XML element
# (one-to-one or composition) or array of child elements (one-to-many or
# aggregation) of this type. Default is one-to-one. Use :array option for one-to-many, or
# simply pass the class in an array.
#
# Composition example:
#
#
# Pragmatic Bookshelf
#
#
#
# Can be mapped using the following code:
# class Book
# xml_reader :publisher, Publisher
# end
#
# Aggregation example:
#
#
#
#
#
#
#
# Can be mapped using the following code:
# class Library
# xml_reader :books, [Book], :in => "books"
# end
#
# If you don't have the tag to wrap around the list of tags:
#
# Ruby books
#
#
#
#
# You can skip the wrapper argument:
# xml_reader :books, [Book]
#
# == Blocks
# You may also pass a block which manipulates the associated parsed value.
#
# class Muffins
# include ROXML
#
# xml_reader(:count, :from => 'bakers_dozens') {|val| val.to_i * 13 }
# end
#
# For hash types, the block recieves the key and value as arguments, and they should
# be returned as an array of [key, value]
#
# === Block Shorthands
#
# Alternatively, you may use block shorthands to specify common coercions, such that:
#
# xml_reader :count, :as => Integer
#
# is equivalent to:
#
# xml_reader(:count) {|val| Integer(val) }
#
# Block shorthands :float, Float, :integer and Integer are currently available,
# but only for non-Hash declarations.
#
# == Other options
# [:from] The name by which the xml value will be found, either an attribute or tag name in XML. Default is sym, or the singular form of sym, in the case of arrays and hashes.
# [:as] :cdata for character data; :integer, Integer, :float, Float to coerce to Integer or Float respectively
# [:in] An optional name of a wrapping tag for this XML accessor
# [:else] Default value for attribute, if missing
# [:required] If true, throws RequiredElementMissing when the element isn't present
#
def xml(sym, writable = false, type_and_or_opts = :text, opts = nil, &block)
opts = Opts.new(sym, *[type_and_or_opts, opts].compact, &block)
tag_refs << case opts.type
when :attr then XMLAttributeRef
when :content then XMLTextRef
when :text then XMLTextRef
when :hash then XMLHashRef
when Symbol then raise ArgumentError, "Invalid type argument #{opts.type}"
else XMLObjectRef
end.new(sym, opts)
add_accessor(sym, writable, opts.array?, opts.default)
end
# Declares a read-only xml reference. See xml for details.
def xml_reader(sym, type_and_or_opts = :text, opts = nil, &block)
xml sym, false, type_and_or_opts, opts, &block
end
# Declares a writable xml reference. See xml for details.
def xml_accessor(sym, type_and_or_opts = :text, opts = nil, &block)
xml sym, true, type_and_or_opts, opts, &block
end
# On parse, call the target object's initialize function with the listed arguments
def xml_construct(*args)
if missing_tag = args.detect {|arg| !tag_refs.map(&:name).include?(arg.to_s) }
raise ArgumentError, "All construction tags must be declared first using xml, " +
"xml_reader, or xml_accessor. #{missing_tag} is missing. " +
tag_refs.map(&:name).join(', ') + ' are declared.'
end
@xml_construction_args = args
end
private
def assert_accessor(name)
@tag_accessors ||= []
raise "Accessor #{name} is already defined as XML accessor in class #{self}" if @tag_accessors.include?(name)
@tag_accessors << name
end
def add_accessor(name, writable, as_array, default = nil)
assert_accessor(name)
unless instance_methods.include?(name)
default ||= Array.new if as_array
define_method(name) do
val = instance_variable_get("@#{name}")
if val.nil?
val = default.duplicable? ? default.dup : default
instance_variable_set("@#{name}", val)
end
val
end
end
if writable && !instance_methods.include?("#{name}=")
define_method("#{name}=") do |v|
instance_variable_set("@#{name}", v)
end
end
end
end
module Accessors
def xml_construction_args # :nodoc:
@xml_construction_args ||= []
end
# Returns the tag name (also known as xml_name) of the class.
# If no tag name is set with xml_name method, returns default class name
# in lowercase.
def tag_name
@tag_name ||= name.split('::').last.downcase
end
# Returns array of internal reference objects, such as attributes
# and composed XML objects
def tag_refs
@xml_refs ||= []
end
end
module Operations
#
# Creates a new Ruby object from XML using mapping information
# annotated in the class.
#
# The input data is either an XML::Node or a String representing
# the XML document.
#
# Example
# book = Book.from_xml(File.read("book.xml"))
# or
# book = Book.from_xml("Beyond Java")
#
# See also: xml_construct
#
def from_xml(data)
xml = (data.kind_of?(XML::Node) ? data : XML::Parser.parse(data).root)
unless xml_construction_args.empty?
args = xml_construction_args.map do |arg|
tag_refs.find {|ref| ref.name == arg.to_s }
end.map {|ref| ref.value(xml) }
new(*args)
else
returning allocate do |inst|
tag_refs.each do |ref|
ref.populate(xml, inst)
end
end
end
end
# Deprecated in favor of #from_xml
def parse(data)
ActiveSupport::Deprecation.warn '#parse has been deprecated, please use #from_xml instead'
from_xml(data)
end
end
end
end