package XML::SimpleObject::LibXML; use strict; use XML::LibXML; our $VERSION = '0.53'; sub attributes { my $self = shift; my $name = shift; my %attributes; my @attrs = $self->{_DOM}->getAttributes; foreach my $attribute (@attrs) { $attributes{$attribute->getName} = $attribute->value; } return %attributes; } sub attribute { my $self = shift; my $name = shift; my ($found) = $self->{_DOM}->findnodes("\@$name"); if ($found) { return $found->value; } } sub value { my $node = shift; my ($found) = $node->{_DOM}->findnodes("text()"); if ($found) { return $found->getData(); } } sub name { $_[0]->{_DOM}->getName; } sub child { my $self = shift; my $tag = shift; if (ref $self->{_DOM} eq "XML::LibXML::Document") { my $node = new XML::SimpleObject::LibXML ($self->{_DOM}->documentElement()); return $node; } else { my ($element) = $self->{_DOM}->getElementsByTagName($tag); return unless ($element); my $node = new XML::SimpleObject::LibXML ($element); return $node; } } sub children_names { my $self = shift; my @elements; foreach my $node ($self->{_DOM}->getChildnodes) { next if ($node->nodeType == 3); push @elements, $node->getName; } return @elements; } sub children { my $self = shift; my $tag = shift; if (ref $self->{_DOM} eq "XML::LibXML::Document") { my $node = new XML::SimpleObject::LibXML ($self->{_DOM}->documentElement()); return $node; } else { if ($tag) { my @nodelist; foreach my $node ($self->{_DOM}->getElementsByTagName($tag)) { next if ($node->nodeType == 3); push @nodelist, new XML::SimpleObject::LibXML ($node); } return @nodelist; } else { my @nodelist; foreach my $node ($self->{_DOM}->getChildnodes()) { next if ($node->nodeType == 3); push @nodelist, new XML::SimpleObject::LibXML ($node); } return @nodelist; } } } sub new { my $class = shift; if (ref($_[0]) =~ /^XML\:\:LibXML/) { my $self = {}; bless ($self,$class); $self->{_DOM} = $_[0]; return $self; } else { my %args = @_; my $parser = new XML::LibXML; my $dom = $parser->parse_string($args{XML}); my $self = {}; bless ($self,$class); $self->{_NAME} = ""; $self->{_DOM} = $dom; return $self; } } 1; __END__ =head1 NAME XML::SimpleObject::LibXML - Perl extension allowing a simple(r) object representation of an XML::LibXML DOM object. =head1 SYNOPSIS use XML::SimpleObject::LibXML; # Construct with the key/value pairs as argument; this will create its # own XML::LibXML object. my $xmlobj = new XML::SimpleObject(XML => $XML); # ... or construct with the parsed tree as the only argument, having to # create the XML::Parser object separately. my $parser = new XML::LibXML; my $dom = $parser->parse_file($file); # or $parser->parse_string($xml); my $xmlobj = new XML::SimpleObject::LibXML ($dom); my $filesobj = $xmlobj->child("files")->child("file"); $filesobj->name; $filesobj->value; $filesobj->attribute("type"); %attributes = $filesobj->attributes; @children = $filesobj->children; @some_children = $filesobj->children("some"); @children_names = $filesobj->children_names; =head1 DESCRIPTION This is a short and simple class allowing simple object access to a parsed XML::LibXML tree, with methods for fetching children and attributes in as clean a manner as possible. My apologies for further polluting the XML:: space; this is a small and quick module, with easy and compact usage. Some will rightfully question placing another interface over the DOM methods provided by XML::LibXML, but my experience is that people appreciate the total simplicity provided by this module, despite its limitations. =head1 USAGE =item $xmlobj = new XML::SimpleObject::LibXML($parser->parse_string($XML)) $parser is an XML::LibXML object. After creating $xmlobj, this object can now be used to browse the XML tree with the following methods. =item $xmlobj->child('NAME') This will return a new XML::SimpleObject::LibXML object using the child element NAME. =item $xmlobj->children('NAME') Called with an argument NAME, children() will return an array of XML::SimpleObject::LibXML objects of element NAME. Thus, if $xmlobj represents the top-level XML element, 'children' will return an array of all elements directly below the top-level that have the element name NAME. =item $xmlobj->children Called without arguments, 'children()' will return an array of XML::SimpleObjects::LibXML objects for all children elements of $xmlobj. Unlike XML::SimpleObject, XML::SimpleObject::LibXML retains the order of these children. =item $xmlobj->children_names This will return an array of all the names of child elements for $xmlobj. You can use this to step through all the children of a given element (see EXAMPLES), although multiple elements of the same name will not be identified. Use 'children()' instead. =item $xmlobj->value If the element represented by $xmlobj contains any PCDATA, this method will return that text data. =item $xmlobj->attribute('NAME') This returns the text for an attribute NAME of the XML element represented by $xmlobj. =item $xmlobj->attributes This returns a hash of key/value pairs for all elements in element $xmlobj. =head1 EXAMPLES Given this XML document: /etc/dosemu.conf dosemu.conf-drdos703.eval /etc/passwd 948 You can then interpret the tree as follows: my $parser = new XML::LibXML; my $xmlobj = new XML::SimpleObject::LibXML ($parser->parse_string($XML)); print "Files: \n"; foreach my $element ($xmlobj->child("files")->children("file")) { print " filename: " . $element->child("name")->value . "\n"; if ($element->attribute("type")) { print " type: " . $element->attribute("type") . "\n"; } print " bytes: " . $element->child("bytes")->value . "\n"; } This will output: Files: filename: /etc/dosemu.conf type: symlink bytes: 20 filename: /etc/passwd bytes: 948 You can use 'children()' without arguments to step through all children of a given element: my $filesobj = $xmlobj->child("files")->child("file"); foreach my $child ($filesobj->children) { print "child: ", $child->name, ": ", $child->value, "\n"; } For the tree above, this will output: child: bytes: 20 child: dest: dosemu.conf-drdos703.eval child: name: /etc/dosemu.conf Using 'children_names()', you can step through all children for a given element: my $filesobj = $xmlobj->child("files"); foreach my $childname ($filesobj->children_names) { print "$childname has children: "; print join (", ", $filesobj->child($childname)->children_names), "\n"; } This will print: file has children: bytes, dest, name By always using 'children()', you can step through each child object, retrieving them with 'child()'. =head1 AUTHOR Dan Brian =head1 SEE ALSO perl(1), XML::SimpleObject, XML::Parser, XML::LibXML. =cut