| Data-Dump-XML documentation | Contained in the Data-Dump-XML distribution. |
Data::Dump::XML::Parser - Restore data dumped by Data::DumpXML
use Data::Dump::XML::Parser; my $p = Data::Dump::XML::Parser->new; my $data = $p->parse_uri(shift || "test.xml");
The Data::Dump::XML::Parser is an XML::LibXML::SAX subclass that
will recreate the data structure from the XML document produced by
Data::Dump::XML. The parserfile() method returns a reference
to an array of the values dumped.
The constructor method new() takes a single additional argument to
that of XML::LibXML::SAX :
A subroutine that is invoked for blessing of restored objects. The
subroutine is invoked with two arguments; a reference to the object
and a string containing the class name. If not provided the built in
bless function is used.
For situations where the input file cannot necessarily be trusted and blessing arbitrary Classes might give the ability of malicious input to exploit the DESTROY methods of modules used by the code it is a good idea to provide an noop blesser:
my $p = Data::Dump::XML::Parser->new(Blesser => sub {});
The Data::Dump::XML::Parser module is written by Ivan
Baktsheev <dot.and.thing@gmail.com>, with support Ivan Kurmanov
<kurmanov@openlib.og>.
Based on Data::DumpXML::Parser written by Gisle Aas
<gisle@aas.no>.
Copyright 2004-2009 Ivan Baktsheev Copyright 2003 Ivan Baktsheev, Ivan Kurmanov Copyright 2001 Gisle Aas.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
| Data-Dump-XML documentation | Contained in the Data-Dump-XML distribution. |
package Data::Dump::XML::Parser; # # Once upon a time this file was a part of ACIS software, # http://acis.openlib.org/ # # Description: # # Parse an Data::Dump::XML-dumped XML string and recreate the # data structure. # # This module is tightly related to Data::Dump::XML, which is # based on Data::DumpXML, and it is accordingly based on # Data::DumpXML::Parser. # # Copyright 2004-2009 Ivan Baktsheev # Copyright 2003 Ivan Baktsheev, Ivan Kurmanov # Copyright 1998-2003 Gisle Aas. # Copyright 1996-1998 Gurusamy Sarathy. # # XXX use of GNU GPL here is questionable. # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License, version 2, as # published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA # # --- # $Id: Parser.pm,v 1.16 2009/06/07 20:22:56 apla Exp $ # --- use Class::Easy; use base qw(XML::LibXML::SAX); use Data::Dump::XML; our $VERSION = 1.19; # require XSLoader; # XSLoader::load ('Data::Dump::XML::Parser', $VERSION); #*characters = \&Data::Dump::XML::characters; sub new { my($class, %arg) = @_; Data::Dump::XML->new unless defined $Data::Dump::XML::INSTANCE; $arg{defaults} = {%$Data::Dump::XML::defaults}; return bless \%arg, $class; } sub start_document { my $p = shift; # real restored data $p->{data} = undef; # data stack. used for control child/parent axe $p->{stack} = []; push @{$p->{stack}}, \$p->{data}; # current depth $p->{depth} = 0; } sub start_element { my ($p, $element) = @_; my $d = $p->{defaults}; my %attr = map {$_->{LocalName} => $_->{Value}} values %{$element->{Attributes}}; my $tag = $element->{LocalName}; my $depth = \$p->{depth}; $$depth++; $p->{max_depth} = $$depth; if ($$depth == 1) { $d->{'root_name'} = $tag; foreach (qw(ref_element hash_element array_element empty_array empty_hash undef key_as_hash_element at_key_as_attribute hash_element_attribute_name) ) { $d->{$_} = delete $attr{"_$_"} if exists $attr{"_$_"}; } } my $key_as_hash_element = $d->{'key_as_hash_element'}; my $root_name = $d->{'root_name'}; my $ref_element = $d->{'ref_element'}; my $array_element = $d->{'array_element'}; my $hash_element = $d->{'hash_element'}; my $empty_array = $d->{'empty_array'}; my $undef = $d->{'undef'}; my $empty_hash = $d->{'empty_hash'}; my $blesser; $blesser = $p->{Blesser} if (exists $p->{Blesser} and ref($blesser) eq "CODE"); my $parent_attr = $p->{attr}->[-1]; my $parent_class = delete $parent_attr->{_class}; my $parent_id = delete $parent_attr->{_id}; my $ref = $p->{'stack'}->[-1]; #my $defined_parent = 0; #$defined_parent = 1 # if ref $p->{'stack'}->[-1] eq 'SCALAR' # and not defined ${$p->{'stack'}->[-1]}; push (@{$p->{'attr'}}, \%attr); if ($$depth == 1) {# and not defined $ref) { # root element } elsif ($tag eq $array_element) { #$$ref = [] # if $defined_parent; ### check the data type die "'$tag' elements only appear in list elements" if defined $$ref and Data::Dump::XML::reftype ($$ref) ne 'ARRAY'; push @{$$ref}, undef; push @{$p->{'stack'}}, \($$ref->[-1]); $blesser ? &$blesser ($$ref, $parent_class) : bless ($$ref, $parent_class) if defined $parent_class; } elsif ($tag eq $ref_element) { my $value = undef; $$ref = \$value; $$ref = ${$p->{'id'}->[$attr{'to'}]} if (defined $attr{'to'}); push @{$p->{'stack'}}, $$ref; } elsif ($tag eq $undef) { $$ref = undef; push @{$p->{'stack'}}, undef; } elsif ($tag eq $empty_hash) { $$ref = {}; push @{$p->{'stack'}}, undef; } elsif ($tag eq $empty_array) { $$ref = []; push @{$p->{'stack'}}, undef; } elsif ($key_as_hash_element or ($tag eq $hash_element and exists $attr{$d->{hash_element_attribute_name}})) { #$$ref = {} # if $defined_parent; my $key = $tag; $key = delete $attr{$d->{hash_element_attribute_name}} if exists $attr{$d->{hash_element_attribute_name}}; die "hash element '$key' must appear in hash context" if defined $$ref and Data::Dump::XML::reftype ($$ref) ne 'HASH'; unless (defined $$ref) { # copy all attributes except _* foreach my $k (keys %$parent_attr) { # next if substr ($k, 0, 1) eq '_'; $$ref->{"\@$k"} = $parent_attr->{$k}; } } die "hash element '$key' already present" if exists $$ref->{$key}; $$ref->{$key} = undef; push @{$p->{stack}}, \(${$ref}->{$key}); $blesser ? &$blesser ($$ref, $parent_class) : bless ($$ref, $parent_class) if defined $parent_class; } else { warn "found unknown element $tag"; } # mix of pcdata and elements not allowed, ignore chars $p->{char} = ''; $p->{id}->[$parent_id] = $ref if ($parent_id); } sub characters { my ($p, $str) = @_; $p->{'char'} .= $str->{'Data'} if defined $str->{'Data'}; } sub end_element { my ($p, $element) = @_; my $d = $p->{defaults}; my $tag = $element->{'LocalName'}; my $depth = \$p->{depth}; my $str = $p->{'char'}; my $ref = pop @{$p->{'stack'}}; $p->{'char'} = ''; my $attr = pop @{$p->{'attr'}}; my $attributed_keys = {map {$_ => $attr->{$_}} grep {!/^_/} keys %$attr}; my $has_attrs = scalar keys %$attributed_keys; if( $$depth < $p->{max_depth}) { #print ' 'x $$depth, "- this element had children\n"; } else { # here processing for empty tags #my $key = $tag; #$key = delete $attr{$d->{hash_element_attribute_name}} # if exists $attr{$d->{hash_element_attribute_name}}; if ( # ($tag eq $d->{array_element} or $tag eq $d->{hash_element}) and $has_attrs ) { $$ref->{'#text'} = $str if defined $str and $str ne ''; foreach my $k (keys %$attributed_keys) { $$ref->{"\@$k"} = $attributed_keys->{$k}; } } elsif ($tag ne $d->{'undef'}) { if ($tag eq $d->{ref_element} and $attr->{'to'}) { # print "'", $p->{'attr'}->[0]->{'to'}, "'\n"; # my $place = $p->{'attr'}->[0]->{'to'}; # # $$ref = ${$p->{'id'}->[$place]} # if (defined $place); } else { #print ' 'x $$depth, "element '$tag' holds a string value ('$str')\n"; $$ref = $str; } } my $class = $attr->{_class}; my $blesser; $blesser = $p->{Blesser} if (exists $p->{Blesser} and ref ($blesser) eq "CODE"); $blesser ? &$blesser ($$ref, $class) : bless ($$ref, $class) if defined $class and ref $$ref; } $$depth--; } sub end_document { my $p = shift; my $data = $p->{'data'}; return $data; } 1; __END__