| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package XML::Simple; |
|
2
|
|
|
|
|
|
|
$XML::Simple::VERSION = '2.22'; |
|
3
|
|
|
|
|
|
|
=head1 NAME |
|
4
|
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
XML::Simple - An API for simple XML files |
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
You really don't want to use this module in new code. If you ignore this |
|
10
|
|
|
|
|
|
|
warning and use it anyway, the C mode will save you a little pain. |
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
use XML::Simple qw(:strict); |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
my $ref = XMLin([] [, ]); |
|
15
|
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
my $xml = XMLout($hashref [, ]); |
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
Or the object oriented way: |
|
19
|
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
require XML::Simple qw(:strict); |
|
21
|
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
my $xs = XML::Simple->new([]); |
|
23
|
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
my $ref = $xs->XMLin([] [, ]); |
|
25
|
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
my $xml = $xs->XMLout($hashref [, ]); |
|
27
|
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
(or see L<"SAX SUPPORT"> for 'the SAX way'). |
|
29
|
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
Note, in these examples, the square brackets are used to denote optional items |
|
31
|
|
|
|
|
|
|
not to imply items should be supplied in arrayrefs. |
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
=cut |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
# See after __END__ for more POD documentation |
|
36
|
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
# Load essentials here, other modules loaded on demand later |
|
39
|
|
|
|
|
|
|
|
|
40
|
12
|
|
|
12
|
|
405441
|
use strict; |
|
|
12
|
|
|
|
|
28
|
|
|
|
12
|
|
|
|
|
319
|
|
|
41
|
12
|
|
|
12
|
|
60
|
use warnings; |
|
|
12
|
|
|
|
|
21
|
|
|
|
12
|
|
|
|
|
1466
|
|
|
42
|
12
|
|
|
12
|
|
71
|
use warnings::register; |
|
|
12
|
|
|
|
|
25
|
|
|
|
12
|
|
|
|
|
1679
|
|
|
43
|
12
|
|
|
12
|
|
68
|
use Carp; |
|
|
12
|
|
|
|
|
26
|
|
|
|
12
|
|
|
|
|
803
|
|
|
44
|
12
|
|
|
12
|
|
67
|
use Scalar::Util qw(); |
|
|
12
|
|
|
|
|
17
|
|
|
|
12
|
|
|
|
|
379
|
|
|
45
|
|
|
|
|
|
|
require Exporter; |
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
############################################################################## |
|
49
|
|
|
|
|
|
|
# Define some constants |
|
50
|
|
|
|
|
|
|
# |
|
51
|
|
|
|
|
|
|
|
|
52
|
12
|
|
|
12
|
|
53
|
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK $PREFERRED_PARSER); |
|
|
12
|
|
|
|
|
21
|
|
|
|
12
|
|
|
|
|
111218
|
|
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
@ISA = qw(Exporter); |
|
55
|
|
|
|
|
|
|
@EXPORT = qw(XMLin XMLout); |
|
56
|
|
|
|
|
|
|
@EXPORT_OK = qw(xml_in xml_out); |
|
57
|
|
|
|
|
|
|
$PREFERRED_PARSER = undef; |
|
58
|
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
my %StrictMode = (); |
|
60
|
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
my @KnownOptIn = qw(keyattr keeproot forcecontent contentkey noattr |
|
62
|
|
|
|
|
|
|
searchpath forcearray cache suppressempty parseropts |
|
63
|
|
|
|
|
|
|
grouptags nsexpand datahandler varattr variables |
|
64
|
|
|
|
|
|
|
normalisespace normalizespace valueattr strictmode); |
|
65
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
my @KnownOptOut = qw(keyattr keeproot contentkey noattr |
|
67
|
|
|
|
|
|
|
rootname xmldecl outputfile noescape suppressempty |
|
68
|
|
|
|
|
|
|
grouptags nsexpand handler noindent attrindent nosort |
|
69
|
|
|
|
|
|
|
valueattr numericescape strictmode); |
|
70
|
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
my @DefKeyAttr = qw(name key id); |
|
72
|
|
|
|
|
|
|
my $DefRootName = qq(opt); |
|
73
|
|
|
|
|
|
|
my $DefContentKey = qq(content); |
|
74
|
|
|
|
|
|
|
my $DefXmlDecl = qq(); |
|
75
|
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
my $xmlns_ns = 'http://www.w3.org/2000/xmlns/'; |
|
77
|
|
|
|
|
|
|
my $bad_def_ns_jcn = '{' . $xmlns_ns . '}'; # LibXML::SAX workaround |
|
78
|
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
############################################################################## |
|
81
|
|
|
|
|
|
|
# Globals for use by caching routines |
|
82
|
|
|
|
|
|
|
# |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
my %MemShareCache = (); |
|
85
|
|
|
|
|
|
|
my %MemCopyCache = (); |
|
86
|
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
############################################################################## |
|
89
|
|
|
|
|
|
|
# Wrapper for Exporter - handles ':strict' |
|
90
|
|
|
|
|
|
|
# |
|
91
|
|
|
|
|
|
|
|
|
92
|
|
|
|
|
|
|
sub import { |
|
93
|
|
|
|
|
|
|
# Handle the :strict tag |
|
94
|
|
|
|
|
|
|
|
|
95
|
11
|
|
|
11
|
|
380
|
my($calling_package) = caller(); |
|
96
|
11
|
100
|
|
|
|
82
|
_strict_mode_for_caller(1) if grep(/^:strict$/, @_); |
|
97
|
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
# Pass everything else to Exporter.pm |
|
99
|
|
|
|
|
|
|
|
|
100
|
11
|
|
|
|
|
45
|
@_ = grep(!/^:strict$/, @_); |
|
101
|
11
|
|
|
|
|
19630
|
goto &Exporter::import; |
|
102
|
|
|
|
|
|
|
} |
|
103
|
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
############################################################################## |
|
106
|
|
|
|
|
|
|
# Constructor for optional object interface. |
|
107
|
|
|
|
|
|
|
# |
|
108
|
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
sub new { |
|
110
|
271
|
|
|
271
|
0
|
23341
|
my $class = shift; |
|
111
|
|
|
|
|
|
|
|
|
112
|
271
|
100
|
|
|
|
826
|
if(@_ % 2) { |
|
113
|
1
|
|
|
|
|
212
|
croak "Default options must be name=>value pairs (odd number supplied)"; |
|
114
|
|
|
|
|
|
|
} |
|
115
|
|
|
|
|
|
|
|
|
116
|
270
|
|
|
|
|
398
|
my %known_opt; |
|
117
|
270
|
|
|
|
|
4108
|
@known_opt{@KnownOptIn, @KnownOptOut} = (); |
|
118
|
|
|
|
|
|
|
|
|
119
|
270
|
|
|
|
|
609
|
my %raw_opt = @_; |
|
120
|
|
|
|
|
|
|
$raw_opt{strictmode} = _strict_mode_for_caller() |
|
121
|
270
|
50
|
|
|
|
1005
|
unless exists $raw_opt{strictmode}; |
|
122
|
270
|
|
|
|
|
688
|
my %def_opt; |
|
123
|
270
|
|
|
|
|
966
|
while(my($key, $val) = each %raw_opt) { |
|
124
|
309
|
|
|
|
|
591
|
my $lkey = lc($key); |
|
125
|
309
|
|
|
|
|
615
|
$lkey =~ s/_//g; |
|
126
|
309
|
100
|
|
|
|
971
|
croak "Unrecognised option: $key" unless(exists($known_opt{$lkey})); |
|
127
|
308
|
|
|
|
|
1381
|
$def_opt{$lkey} = $val; |
|
128
|
|
|
|
|
|
|
} |
|
129
|
269
|
|
|
|
|
726
|
my $self = { def_opt => \%def_opt }; |
|
130
|
|
|
|
|
|
|
|
|
131
|
269
|
|
|
|
|
1565
|
return(bless($self, $class)); |
|
132
|
|
|
|
|
|
|
} |
|
133
|
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
############################################################################## |
|
136
|
|
|
|
|
|
|
# Sub: _strict_mode_for_caller() |
|
137
|
|
|
|
|
|
|
# |
|
138
|
|
|
|
|
|
|
# Gets or sets the XML::Simple :strict mode flag for the calling namespace. |
|
139
|
|
|
|
|
|
|
# Walks back through call stack to find the calling namespace and sets the |
|
140
|
|
|
|
|
|
|
# :strict mode flag for that namespace if an argument was supplied and returns |
|
141
|
|
|
|
|
|
|
# the flag value if not. |
|
142
|
|
|
|
|
|
|
# |
|
143
|
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
sub _strict_mode_for_caller { |
|
145
|
271
|
|
|
271
|
|
457
|
my $set_mode = @_; |
|
146
|
271
|
|
|
|
|
359
|
my $frame = 1; |
|
147
|
271
|
|
|
|
|
2292
|
while(my($package) = caller($frame++)) { |
|
148
|
757
|
100
|
|
|
|
4404
|
next if $package eq 'XML::Simple'; |
|
149
|
271
|
100
|
|
|
|
582
|
$StrictMode{$package} = 1 if $set_mode; |
|
150
|
271
|
|
|
|
|
881
|
return $StrictMode{$package}; |
|
151
|
|
|
|
|
|
|
} |
|
152
|
0
|
|
|
|
|
0
|
return(0); |
|
153
|
|
|
|
|
|
|
} |
|
154
|
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
############################################################################## |
|
157
|
|
|
|
|
|
|
# Sub: _get_object() |
|
158
|
|
|
|
|
|
|
# |
|
159
|
|
|
|
|
|
|
# Helper routine called from XMLin() and XMLout() to create an object if none |
|
160
|
|
|
|
|
|
|
# was provided. Note, this routine does mess with the caller's @_ array. |
|
161
|
|
|
|
|
|
|
# |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
sub _get_object { |
|
164
|
446
|
|
|
446
|
|
643
|
my $self; |
|
165
|
446
|
100
|
100
|
|
|
4803
|
if($_[0] and UNIVERSAL::isa($_[0], 'XML::Simple')) { |
|
166
|
203
|
|
|
|
|
338
|
$self = shift; |
|
167
|
|
|
|
|
|
|
} |
|
168
|
|
|
|
|
|
|
else { |
|
169
|
243
|
|
|
|
|
791
|
$self = XML::Simple->new(); |
|
170
|
|
|
|
|
|
|
} |
|
171
|
|
|
|
|
|
|
|
|
172
|
446
|
|
|
|
|
839
|
return $self; |
|
173
|
|
|
|
|
|
|
} |
|
174
|
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
|
|
176
|
|
|
|
|
|
|
############################################################################## |
|
177
|
|
|
|
|
|
|
# Sub/Method: XMLin() |
|
178
|
|
|
|
|
|
|
# |
|
179
|
|
|
|
|
|
|
# Exported routine for slurping XML into a hashref - see pod for info. |
|
180
|
|
|
|
|
|
|
# |
|
181
|
|
|
|
|
|
|
# May be called as object method or as a plain function. |
|
182
|
|
|
|
|
|
|
# |
|
183
|
|
|
|
|
|
|
# Expects one arg for the source XML, optionally followed by a number of |
|
184
|
|
|
|
|
|
|
# name => value option pairs. |
|
185
|
|
|
|
|
|
|
# |
|
186
|
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
sub XMLin { |
|
188
|
180
|
|
|
180
|
1
|
10239130
|
my $self = &_get_object; # note, @_ is passed implicitly |
|
189
|
|
|
|
|
|
|
|
|
190
|
180
|
|
|
|
|
339
|
my $target = shift; |
|
191
|
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
# Work out whether to parse a string, a file or a filehandle |
|
194
|
|
|
|
|
|
|
|
|
195
|
180
|
100
|
|
|
|
1411
|
if(not defined $target) { |
|
|
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
196
|
1
|
|
|
|
|
4
|
return $self->parse_file(undef, @_); |
|
197
|
|
|
|
|
|
|
} |
|
198
|
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
elsif($target eq '-') { |
|
200
|
2
|
|
|
|
|
12
|
local($/) = undef; |
|
201
|
2
|
|
|
|
|
59
|
$target = ; |
|
202
|
2
|
|
|
|
|
10
|
return $self->parse_string(\$target, @_); |
|
203
|
|
|
|
|
|
|
} |
|
204
|
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
elsif(my $type = ref($target)) { |
|
206
|
2
|
50
|
|
|
|
10
|
if($type eq 'SCALAR') { |
|
207
|
0
|
|
|
|
|
0
|
return $self->parse_string($target, @_); |
|
208
|
|
|
|
|
|
|
} |
|
209
|
|
|
|
|
|
|
else { |
|
210
|
2
|
|
|
|
|
11
|
return $self->parse_fh($target, @_); |
|
211
|
|
|
|
|
|
|
} |
|
212
|
|
|
|
|
|
|
} |
|
213
|
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
elsif($target =~ m{<.*?>}s) { |
|
215
|
147
|
|
|
|
|
450
|
return $self->parse_string(\$target, @_); |
|
216
|
|
|
|
|
|
|
} |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
else { |
|
219
|
28
|
|
|
|
|
141
|
return $self->parse_file($target, @_); |
|
220
|
|
|
|
|
|
|
} |
|
221
|
|
|
|
|
|
|
} |
|
222
|
|
|
|
|
|
|
|
|
223
|
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
############################################################################## |
|
225
|
|
|
|
|
|
|
# Sub/Method: parse_file() |
|
226
|
|
|
|
|
|
|
# |
|
227
|
|
|
|
|
|
|
# Same as XMLin, but only parses from a named file. |
|
228
|
|
|
|
|
|
|
# |
|
229
|
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
sub parse_file { |
|
231
|
29
|
|
|
29
|
1
|
90
|
my $self = &_get_object; # note, @_ is passed implicitly |
|
232
|
|
|
|
|
|
|
|
|
233
|
29
|
|
|
|
|
71
|
my $filename = shift; |
|
234
|
|
|
|
|
|
|
|
|
235
|
29
|
|
|
|
|
147
|
$self->handle_options('in', @_); |
|
236
|
|
|
|
|
|
|
|
|
237
|
27
|
100
|
|
|
|
96
|
$filename = $self->default_config_file if not defined $filename; |
|
238
|
|
|
|
|
|
|
|
|
239
|
27
|
|
|
|
|
58
|
$filename = $self->find_xml_file($filename, @{$self->{opt}->{searchpath}}); |
|
|
27
|
|
|
|
|
143
|
|
|
240
|
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
# Check cache for previous parse |
|
242
|
|
|
|
|
|
|
|
|
243
|
24
|
100
|
|
|
|
110
|
if($self->{opt}->{cache}) { |
|
244
|
17
|
|
|
|
|
45
|
foreach my $scheme (@{$self->{opt}->{cache}}) { |
|
|
17
|
|
|
|
|
76
|
|
|
245
|
17
|
|
|
|
|
50
|
my $method = 'cache_read_' . $scheme; |
|
246
|
17
|
|
|
|
|
153
|
my $opt = $self->$method($filename); |
|
247
|
17
|
100
|
|
|
|
685
|
return($opt) if($opt); |
|
248
|
|
|
|
|
|
|
} |
|
249
|
|
|
|
|
|
|
} |
|
250
|
|
|
|
|
|
|
|
|
251
|
16
|
|
|
|
|
83
|
my $ref = $self->build_simple_tree($filename, undef); |
|
252
|
|
|
|
|
|
|
|
|
253
|
16
|
100
|
|
|
|
131
|
if($self->{opt}->{cache}) { |
|
254
|
9
|
|
|
|
|
33
|
my $method = 'cache_write_' . $self->{opt}->{cache}->[0]; |
|
255
|
9
|
|
|
|
|
47
|
$self->$method($ref, $filename); |
|
256
|
|
|
|
|
|
|
} |
|
257
|
|
|
|
|
|
|
|
|
258
|
16
|
|
|
|
|
15082
|
return $ref; |
|
259
|
|
|
|
|
|
|
} |
|
260
|
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
|
|
262
|
|
|
|
|
|
|
############################################################################## |
|
263
|
|
|
|
|
|
|
# Sub/Method: parse_fh() |
|
264
|
|
|
|
|
|
|
# |
|
265
|
|
|
|
|
|
|
# Same as XMLin, but only parses from a filehandle. |
|
266
|
|
|
|
|
|
|
# |
|
267
|
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
sub parse_fh { |
|
269
|
2
|
|
|
2
|
1
|
8
|
my $self = &_get_object; # note, @_ is passed implicitly |
|
270
|
|
|
|
|
|
|
|
|
271
|
2
|
|
|
|
|
6
|
my $fh = shift; |
|
272
|
2
|
0
|
|
|
|
10
|
croak "Can't use " . (defined $fh ? qq{string ("$fh")} : 'undef') . |
|
|
|
50
|
|
|
|
|
|
|
273
|
|
|
|
|
|
|
" as a filehandle" unless ref $fh; |
|
274
|
|
|
|
|
|
|
|
|
275
|
2
|
|
|
|
|
18
|
$self->handle_options('in', @_); |
|
276
|
|
|
|
|
|
|
|
|
277
|
2
|
|
|
|
|
10
|
return $self->build_simple_tree(undef, $fh); |
|
278
|
|
|
|
|
|
|
} |
|
279
|
|
|
|
|
|
|
|
|
280
|
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
############################################################################## |
|
282
|
|
|
|
|
|
|
# Sub/Method: parse_string() |
|
283
|
|
|
|
|
|
|
# |
|
284
|
|
|
|
|
|
|
# Same as XMLin, but only parses from a string or a reference to a string. |
|
285
|
|
|
|
|
|
|
# |
|
286
|
|
|
|
|
|
|
|
|
287
|
|
|
|
|
|
|
sub parse_string { |
|
288
|
149
|
|
|
149
|
1
|
325
|
my $self = &_get_object; # note, @_ is passed implicitly |
|
289
|
|
|
|
|
|
|
|
|
290
|
149
|
|
|
|
|
239
|
my $string = shift; |
|
291
|
|
|
|
|
|
|
|
|
292
|
149
|
|
|
|
|
455
|
$self->handle_options('in', @_); |
|
293
|
|
|
|
|
|
|
|
|
294
|
136
|
50
|
|
|
|
498
|
return $self->build_simple_tree(undef, ref $string ? $string : \$string); |
|
295
|
|
|
|
|
|
|
} |
|
296
|
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
############################################################################## |
|
299
|
|
|
|
|
|
|
# Method: default_config_file() |
|
300
|
|
|
|
|
|
|
# |
|
301
|
|
|
|
|
|
|
# Returns the name of the XML file to parse if no filename (or XML string) |
|
302
|
|
|
|
|
|
|
# was provided. |
|
303
|
|
|
|
|
|
|
# |
|
304
|
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
sub default_config_file { |
|
306
|
1
|
|
|
1
|
1
|
2
|
my $self = shift; |
|
307
|
|
|
|
|
|
|
|
|
308
|
1
|
|
|
|
|
6
|
require File::Basename; |
|
309
|
|
|
|
|
|
|
|
|
310
|
1
|
|
|
|
|
67
|
my($basename, $script_dir, $ext) = File::Basename::fileparse($0, '\.[^\.]+'); |
|
311
|
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
# Add script directory to searchpath |
|
313
|
|
|
|
|
|
|
|
|
314
|
1
|
50
|
|
|
|
5
|
if($script_dir) { |
|
315
|
1
|
|
|
|
|
1
|
unshift(@{$self->{opt}->{searchpath}}, $script_dir); |
|
|
1
|
|
|
|
|
5
|
|
|
316
|
|
|
|
|
|
|
} |
|
317
|
|
|
|
|
|
|
|
|
318
|
1
|
|
|
|
|
3
|
return $basename . '.xml'; |
|
319
|
|
|
|
|
|
|
} |
|
320
|
|
|
|
|
|
|
|
|
321
|
|
|
|
|
|
|
|
|
322
|
|
|
|
|
|
|
############################################################################## |
|
323
|
|
|
|
|
|
|
# Method: build_simple_tree() |
|
324
|
|
|
|
|
|
|
# |
|
325
|
|
|
|
|
|
|
# Builds a 'tree' data structure as provided by XML::Parser and then |
|
326
|
|
|
|
|
|
|
# 'simplifies' it as specified by the various options in effect. |
|
327
|
|
|
|
|
|
|
# |
|
328
|
|
|
|
|
|
|
|
|
329
|
|
|
|
|
|
|
sub build_simple_tree { |
|
330
|
154
|
|
|
154
|
1
|
222
|
my $self = shift; |
|
331
|
|
|
|
|
|
|
|
|
332
|
154
|
|
|
|
|
265
|
my $tree = eval { |
|
333
|
154
|
|
|
|
|
441
|
$self->build_tree(@_); |
|
334
|
|
|
|
|
|
|
}; |
|
335
|
154
|
50
|
|
|
|
579
|
Carp::croak("$@XML::Simple called") if $@; |
|
336
|
|
|
|
|
|
|
|
|
337
|
|
|
|
|
|
|
return $self->{opt}->{keeproot} |
|
338
|
|
|
|
|
|
|
? $self->collapse({}, @$tree) |
|
339
|
154
|
100
|
|
|
|
457
|
: $self->collapse(@{$tree->[1]}); |
|
|
149
|
|
|
|
|
606
|
|
|
340
|
|
|
|
|
|
|
} |
|
341
|
|
|
|
|
|
|
|
|
342
|
|
|
|
|
|
|
|
|
343
|
|
|
|
|
|
|
############################################################################## |
|
344
|
|
|
|
|
|
|
# Method: build_tree() |
|
345
|
|
|
|
|
|
|
# |
|
346
|
|
|
|
|
|
|
# This routine will be called if there is no suitable pre-parsed tree in a |
|
347
|
|
|
|
|
|
|
# cache. It parses the XML and returns an XML::Parser 'Tree' style data |
|
348
|
|
|
|
|
|
|
# structure (summarised in the comments for the collapse() routine below). |
|
349
|
|
|
|
|
|
|
# |
|
350
|
|
|
|
|
|
|
# XML::Simple requires the services of another module that knows how to parse |
|
351
|
|
|
|
|
|
|
# XML. If XML::SAX is installed, the default SAX parser will be used, |
|
352
|
|
|
|
|
|
|
# otherwise XML::Parser will be used. |
|
353
|
|
|
|
|
|
|
# |
|
354
|
|
|
|
|
|
|
# This routine expects to be passed a filename as argument 1 or a 'string' as |
|
355
|
|
|
|
|
|
|
# argument 2. The 'string' might be a string of XML (passed by reference to |
|
356
|
|
|
|
|
|
|
# save memory) or it might be a reference to an IO::Handle. (This |
|
357
|
|
|
|
|
|
|
# non-intuitive mess results in part from the way XML::Parser works but that's |
|
358
|
|
|
|
|
|
|
# really no excuse). |
|
359
|
|
|
|
|
|
|
# |
|
360
|
|
|
|
|
|
|
|
|
361
|
|
|
|
|
|
|
sub build_tree { |
|
362
|
154
|
|
|
154
|
0
|
255
|
my $self = shift; |
|
363
|
154
|
|
|
|
|
214
|
my $filename = shift; |
|
364
|
154
|
|
|
|
|
224
|
my $string = shift; |
|
365
|
|
|
|
|
|
|
|
|
366
|
|
|
|
|
|
|
|
|
367
|
154
|
|
|
|
|
234
|
my $preferred_parser = $PREFERRED_PARSER; |
|
368
|
154
|
100
|
|
|
|
376
|
unless(defined($preferred_parser)) { |
|
369
|
147
|
|
50
|
|
|
682
|
$preferred_parser = $ENV{XML_SIMPLE_PREFERRED_PARSER} || ''; |
|
370
|
|
|
|
|
|
|
} |
|
371
|
154
|
50
|
|
|
|
385
|
if($preferred_parser eq 'XML::Parser') { |
|
372
|
0
|
|
|
|
|
0
|
return($self->build_tree_xml_parser($filename, $string)); |
|
373
|
|
|
|
|
|
|
} |
|
374
|
|
|
|
|
|
|
|
|
375
|
154
|
|
|
|
|
214
|
eval { require XML::SAX; }; # We didn't need it until now |
|
|
154
|
|
|
|
|
7518
|
|
|
376
|
154
|
50
|
|
|
|
46001
|
if($@) { # No XML::SAX - fall back to XML::Parser |
|
377
|
0
|
0
|
|
|
|
0
|
if($preferred_parser) { # unless a SAX parser was expressly requested |
|
378
|
0
|
|
|
|
|
0
|
croak "XMLin() could not load XML::SAX"; |
|
379
|
|
|
|
|
|
|
} |
|
380
|
0
|
|
|
|
|
0
|
return($self->build_tree_xml_parser($filename, $string)); |
|
381
|
|
|
|
|
|
|
} |
|
382
|
|
|
|
|
|
|
|
|
383
|
154
|
50
|
|
|
|
389
|
$XML::SAX::ParserPackage = $preferred_parser if($preferred_parser); |
|
384
|
|
|
|
|
|
|
|
|
385
|
154
|
|
|
|
|
1056
|
my $sp = XML::SAX::ParserFactory->parser(Handler => $self); |
|
386
|
|
|
|
|
|
|
|
|
387
|
154
|
|
|
|
|
867929
|
$self->{nocollapse} = 1; |
|
388
|
154
|
|
|
|
|
248
|
my($tree); |
|
389
|
154
|
100
|
|
|
|
360
|
if($filename) { |
|
390
|
16
|
|
|
|
|
144
|
$tree = $sp->parse_uri($filename); |
|
391
|
|
|
|
|
|
|
} |
|
392
|
|
|
|
|
|
|
else { |
|
393
|
138
|
100
|
66
|
|
|
814
|
if(ref($string) && ref($string) ne 'SCALAR') { |
|
394
|
2
|
|
|
|
|
25
|
$tree = $sp->parse_file($string); |
|
395
|
|
|
|
|
|
|
} |
|
396
|
|
|
|
|
|
|
else { |
|
397
|
136
|
|
|
|
|
540
|
$tree = $sp->parse_string($$string); |
|
398
|
|
|
|
|
|
|
} |
|
399
|
|
|
|
|
|
|
} |
|
400
|
|
|
|
|
|
|
|
|
401
|
154
|
|
|
|
|
2219
|
return($tree); |
|
402
|
|
|
|
|
|
|
} |
|
403
|
|
|
|
|
|
|
|
|
404
|
|
|
|
|
|
|
|
|
405
|
|
|
|
|
|
|
############################################################################## |
|
406
|
|
|
|
|
|
|
# Method: build_tree_xml_parser() |
|
407
|
|
|
|
|
|
|
# |
|
408
|
|
|
|
|
|
|
# This routine will be called if XML::SAX is not installed, or if XML::Parser |
|
409
|
|
|
|
|
|
|
# was specifically requested. It takes the same arguments as build_tree() and |
|
410
|
|
|
|
|
|
|
# returns the same data structure (XML::Parser 'Tree' style). |
|
411
|
|
|
|
|
|
|
# |
|
412
|
|
|
|
|
|
|
|
|
413
|
|
|
|
|
|
|
sub build_tree_xml_parser { |
|
414
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
|
415
|
0
|
|
|
|
|
0
|
my $filename = shift; |
|
416
|
0
|
|
|
|
|
0
|
my $string = shift; |
|
417
|
|
|
|
|
|
|
|
|
418
|
|
|
|
|
|
|
|
|
419
|
0
|
|
|
|
|
0
|
eval { |
|
420
|
0
|
|
|
|
|
0
|
local($^W) = 0; # Suppress warning from Expat.pm re File::Spec::load() |
|
421
|
0
|
|
|
|
|
0
|
require XML::Parser; # We didn't need it until now |
|
422
|
|
|
|
|
|
|
}; |
|
423
|
0
|
0
|
|
|
|
0
|
if($@) { |
|
424
|
0
|
|
|
|
|
0
|
croak "XMLin() requires either XML::SAX or XML::Parser"; |
|
425
|
|
|
|
|
|
|
} |
|
426
|
|
|
|
|
|
|
|
|
427
|
0
|
0
|
|
|
|
0
|
if($self->{opt}->{nsexpand}) { |
|
428
|
0
|
|
|
|
|
0
|
carp "'nsexpand' option requires XML::SAX"; |
|
429
|
|
|
|
|
|
|
} |
|
430
|
|
|
|
|
|
|
|
|
431
|
0
|
|
|
|
|
0
|
my $xp = XML::Parser->new(Style => 'Tree', @{$self->{opt}->{parseropts}}); |
|
|
0
|
|
|
|
|
0
|
|
|
432
|
0
|
|
|
|
|
0
|
my($tree); |
|
433
|
0
|
0
|
|
|
|
0
|
if($filename) { |
|
434
|
|
|
|
|
|
|
# $tree = $xp->parsefile($filename); # Changed due to prob w/mod_perl |
|
435
|
0
|
0
|
|
|
|
0
|
open(my $xfh, '<', $filename) || croak qq($filename - $!); |
|
436
|
0
|
|
|
|
|
0
|
$tree = $xp->parse($xfh); |
|
437
|
|
|
|
|
|
|
} |
|
438
|
|
|
|
|
|
|
else { |
|
439
|
0
|
|
|
|
|
0
|
$tree = $xp->parse($$string); |
|
440
|
|
|
|
|
|
|
} |
|
441
|
|
|
|
|
|
|
|
|
442
|
0
|
|
|
|
|
0
|
return($tree); |
|
443
|
|
|
|
|
|
|
} |
|
444
|
|
|
|
|
|
|
|
|
445
|
|
|
|
|
|
|
|
|
446
|
|
|
|
|
|
|
############################################################################## |
|
447
|
|
|
|
|
|
|
# Method: cache_write_storable() |
|
448
|
|
|
|
|
|
|
# |
|
449
|
|
|
|
|
|
|
# Wrapper routine for invoking Storable::nstore() to cache a parsed data |
|
450
|
|
|
|
|
|
|
# structure. |
|
451
|
|
|
|
|
|
|
# |
|
452
|
|
|
|
|
|
|
|
|
453
|
|
|
|
|
|
|
sub cache_write_storable { |
|
454
|
5
|
|
|
5
|
0
|
12
|
my($self, $data, $filename) = @_; |
|
455
|
|
|
|
|
|
|
|
|
456
|
5
|
|
|
|
|
16
|
my $cachefile = $self->storable_filename($filename); |
|
457
|
|
|
|
|
|
|
|
|
458
|
5
|
|
|
|
|
71
|
require Storable; # We didn't need it until now |
|
459
|
|
|
|
|
|
|
|
|
460
|
5
|
50
|
|
|
|
24
|
if ('VMS' eq $^O) { |
|
461
|
0
|
|
|
|
|
0
|
Storable::nstore($data, $cachefile); |
|
462
|
|
|
|
|
|
|
} |
|
463
|
|
|
|
|
|
|
else { |
|
464
|
|
|
|
|
|
|
# If the following line fails for you, your Storable.pm is old - upgrade |
|
465
|
5
|
|
|
|
|
28
|
Storable::lock_nstore($data, $cachefile); |
|
466
|
|
|
|
|
|
|
} |
|
467
|
|
|
|
|
|
|
|
|
468
|
|
|
|
|
|
|
} |
|
469
|
|
|
|
|
|
|
|
|
470
|
|
|
|
|
|
|
|
|
471
|
|
|
|
|
|
|
############################################################################## |
|
472
|
|
|
|
|
|
|
# Method: cache_read_storable() |
|
473
|
|
|
|
|
|
|
# |
|
474
|
|
|
|
|
|
|
# Wrapper routine for invoking Storable::retrieve() to read a cached parsed |
|
475
|
|
|
|
|
|
|
# data structure. Only returns cached data if the cache file exists and is |
|
476
|
|
|
|
|
|
|
# newer than the source XML file. |
|
477
|
|
|
|
|
|
|
# |
|
478
|
|
|
|
|
|
|
|
|
479
|
|
|
|
|
|
|
sub cache_read_storable { |
|
480
|
8
|
|
|
8
|
0
|
19
|
my($self, $filename) = @_; |
|
481
|
|
|
|
|
|
|
|
|
482
|
8
|
|
|
|
|
83
|
my $cachefile = $self->storable_filename($filename); |
|
483
|
|
|
|
|
|
|
|
|
484
|
8
|
100
|
|
|
|
152
|
return unless(-r $cachefile); |
|
485
|
6
|
100
|
|
|
|
195
|
return unless((stat($cachefile))[9] > (stat($filename))[9]); |
|
486
|
|
|
|
|
|
|
|
|
487
|
3
|
|
|
|
|
23
|
require Storable; # We didn't need it until now |
|
488
|
|
|
|
|
|
|
|
|
489
|
3
|
50
|
|
|
|
23
|
if ('VMS' eq $^O) { |
|
490
|
0
|
|
|
|
|
0
|
return(Storable::retrieve($cachefile)); |
|
491
|
|
|
|
|
|
|
} |
|
492
|
|
|
|
|
|
|
else { |
|
493
|
3
|
|
|
|
|
23
|
return(Storable::lock_retrieve($cachefile)); |
|
494
|
|
|
|
|
|
|
} |
|
495
|
|
|
|
|
|
|
|
|
496
|
|
|
|
|
|
|
} |
|
497
|
|
|
|
|
|
|
|
|
498
|
|
|
|
|
|
|
|
|
499
|
|
|
|
|
|
|
############################################################################## |
|
500
|
|
|
|
|
|
|
# Method: storable_filename() |
|
501
|
|
|
|
|
|
|
# |
|
502
|
|
|
|
|
|
|
# Translates the supplied source XML filename into a filename for the storable |
|
503
|
|
|
|
|
|
|
# cached data. A '.stor' suffix is added after stripping an optional '.xml' |
|
504
|
|
|
|
|
|
|
# suffix. |
|
505
|
|
|
|
|
|
|
# |
|
506
|
|
|
|
|
|
|
|
|
507
|
|
|
|
|
|
|
sub storable_filename { |
|
508
|
12
|
|
|
12
|
0
|
362
|
my($self, $cachefile) = @_; |
|
509
|
|
|
|
|
|
|
|
|
510
|
12
|
|
|
|
|
92
|
$cachefile =~ s{(\.xml)?$}{.stor}; |
|
511
|
12
|
|
|
|
|
43
|
return $cachefile; |
|
512
|
|
|
|
|
|
|
} |
|
513
|
|
|
|
|
|
|
|
|
514
|
|
|
|
|
|
|
|
|
515
|
|
|
|
|
|
|
############################################################################## |
|
516
|
|
|
|
|
|
|
# Method: cache_write_memshare() |
|
517
|
|
|
|
|
|
|
# |
|
518
|
|
|
|
|
|
|
# Takes the supplied data structure reference and stores it away in a global |
|
519
|
|
|
|
|
|
|
# hash structure. |
|
520
|
|
|
|
|
|
|
# |
|
521
|
|
|
|
|
|
|
|
|
522
|
|
|
|
|
|
|
sub cache_write_memshare { |
|
523
|
2
|
|
|
2
|
0
|
6
|
my($self, $data, $filename) = @_; |
|
524
|
|
|
|
|
|
|
|
|
525
|
2
|
|
|
|
|
10
|
$MemShareCache{$filename} = [time(), $data]; |
|
526
|
|
|
|
|
|
|
} |
|
527
|
|
|
|
|
|
|
|
|
528
|
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
############################################################################## |
|
530
|
|
|
|
|
|
|
# Method: cache_read_memshare() |
|
531
|
|
|
|
|
|
|
# |
|
532
|
|
|
|
|
|
|
# Takes a filename and looks in a global hash for a cached parsed version. |
|
533
|
|
|
|
|
|
|
# |
|
534
|
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
sub cache_read_memshare { |
|
536
|
4
|
|
|
4
|
0
|
8
|
my($self, $filename) = @_; |
|
537
|
|
|
|
|
|
|
|
|
538
|
4
|
100
|
|
|
|
20
|
return unless($MemShareCache{$filename}); |
|
539
|
3
|
100
|
|
|
|
63
|
return unless($MemShareCache{$filename}->[0] > (stat($filename))[9]); |
|
540
|
|
|
|
|
|
|
|
|
541
|
2
|
|
|
|
|
9
|
return($MemShareCache{$filename}->[1]); |
|
542
|
|
|
|
|
|
|
|
|
543
|
|
|
|
|
|
|
} |
|
544
|
|
|
|
|
|
|
|
|
545
|
|
|
|
|
|
|
|
|
546
|
|
|
|
|
|
|
############################################################################## |
|
547
|
|
|
|
|
|
|
# Method: cache_write_memcopy() |
|
548
|
|
|
|
|
|
|
# |
|
549
|
|
|
|
|
|
|
# Takes the supplied data structure and stores a copy of it in a global hash |
|
550
|
|
|
|
|
|
|
# structure. |
|
551
|
|
|
|
|
|
|
# |
|
552
|
|
|
|
|
|
|
|
|
553
|
|
|
|
|
|
|
sub cache_write_memcopy { |
|
554
|
2
|
|
|
2
|
0
|
5
|
my($self, $data, $filename) = @_; |
|
555
|
|
|
|
|
|
|
|
|
556
|
2
|
|
|
|
|
19
|
require Storable; # We didn't need it until now |
|
557
|
|
|
|
|
|
|
|
|
558
|
2
|
|
|
|
|
205
|
$MemCopyCache{$filename} = [time(), Storable::dclone($data)]; |
|
559
|
|
|
|
|
|
|
} |
|
560
|
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
|
|
562
|
|
|
|
|
|
|
############################################################################## |
|
563
|
|
|
|
|
|
|
# Method: cache_read_memcopy() |
|
564
|
|
|
|
|
|
|
# |
|
565
|
|
|
|
|
|
|
# Takes a filename and looks in a global hash for a cached parsed version. |
|
566
|
|
|
|
|
|
|
# Returns a reference to a copy of that data structure. |
|
567
|
|
|
|
|
|
|
# |
|
568
|
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
sub cache_read_memcopy { |
|
570
|
4
|
|
|
4
|
0
|
10
|
my($self, $filename) = @_; |
|
571
|
|
|
|
|
|
|
|
|
572
|
4
|
100
|
|
|
|
17
|
return unless($MemCopyCache{$filename}); |
|
573
|
3
|
100
|
|
|
|
64
|
return unless($MemCopyCache{$filename}->[0] > (stat($filename))[9]); |
|
574
|
|
|
|
|
|
|
|
|
575
|
2
|
|
|
|
|
86
|
return(Storable::dclone($MemCopyCache{$filename}->[1])); |
|
576
|
|
|
|
|
|
|
|
|
577
|
|
|
|
|
|
|
} |
|
578
|
|
|
|
|
|
|
|
|
579
|
|
|
|
|
|
|
|
|
580
|
|
|
|
|
|
|
############################################################################## |
|
581
|
|
|
|
|
|
|
# Sub/Method: XMLout() |
|
582
|
|
|
|
|
|
|
# |
|
583
|
|
|
|
|
|
|
# Exported routine for 'unslurping' a data structure out to XML. |
|
584
|
|
|
|
|
|
|
# |
|
585
|
|
|
|
|
|
|
# Expects a reference to a data structure and an optional list of option |
|
586
|
|
|
|
|
|
|
# name => value pairs. |
|
587
|
|
|
|
|
|
|
# |
|
588
|
|
|
|
|
|
|
|
|
589
|
|
|
|
|
|
|
sub XMLout { |
|
590
|
86
|
|
|
86
|
1
|
49469
|
my $self = &_get_object; # note, @_ is passed implicitly |
|
591
|
|
|
|
|
|
|
|
|
592
|
86
|
100
|
|
|
|
582
|
croak "XMLout() requires at least one argument" unless(@_); |
|
593
|
84
|
|
|
|
|
119
|
my $ref = shift; |
|
594
|
|
|
|
|
|
|
|
|
595
|
84
|
|
|
|
|
264
|
$self->handle_options('out', @_); |
|
596
|
|
|
|
|
|
|
|
|
597
|
|
|
|
|
|
|
|
|
598
|
|
|
|
|
|
|
# If namespace expansion is set, XML::NamespaceSupport is required |
|
599
|
|
|
|
|
|
|
|
|
600
|
79
|
100
|
|
|
|
192
|
if($self->{opt}->{nsexpand}) { |
|
601
|
3
|
|
|
|
|
18
|
require XML::NamespaceSupport; |
|
602
|
3
|
|
|
|
|
12
|
$self->{nsup} = XML::NamespaceSupport->new(); |
|
603
|
3
|
|
|
|
|
44
|
$self->{ns_prefix} = 'aaa'; |
|
604
|
|
|
|
|
|
|
} |
|
605
|
|
|
|
|
|
|
|
|
606
|
|
|
|
|
|
|
|
|
607
|
|
|
|
|
|
|
# Wrap top level arrayref in a hash |
|
608
|
|
|
|
|
|
|
|
|
609
|
79
|
100
|
|
|
|
248
|
if(UNIVERSAL::isa($ref, 'ARRAY')) { |
|
610
|
4
|
|
|
|
|
13
|
$ref = { anon => $ref }; |
|
611
|
|
|
|
|
|
|
} |
|
612
|
|
|
|
|
|
|
|
|
613
|
|
|
|
|
|
|
|
|
614
|
|
|
|
|
|
|
# Extract rootname from top level hash if keeproot enabled |
|
615
|
|
|
|
|
|
|
|
|
616
|
79
|
100
|
|
|
|
296
|
if($self->{opt}->{keeproot}) { |
|
|
|
100
|
|
|
|
|
|
|
617
|
2
|
|
|
|
|
8
|
my(@keys) = keys(%$ref); |
|
618
|
2
|
50
|
|
|
|
8
|
if(@keys == 1) { |
|
619
|
2
|
|
|
|
|
6
|
$ref = $ref->{$keys[0]}; |
|
620
|
2
|
|
|
|
|
7
|
$self->{opt}->{rootname} = $keys[0]; |
|
621
|
|
|
|
|
|
|
} |
|
622
|
|
|
|
|
|
|
} |
|
623
|
|
|
|
|
|
|
|
|
624
|
|
|
|
|
|
|
# Ensure there are no top level attributes if we're not adding root elements |
|
625
|
|
|
|
|
|
|
|
|
626
|
|
|
|
|
|
|
elsif($self->{opt}->{rootname} eq '') { |
|
627
|
5
|
100
|
|
|
|
19
|
if(UNIVERSAL::isa($ref, 'HASH')) { |
|
628
|
3
|
|
|
|
|
7
|
my $refsave = $ref; |
|
629
|
3
|
|
|
|
|
4
|
$ref = {}; |
|
630
|
3
|
|
|
|
|
10
|
foreach (keys(%$refsave)) { |
|
631
|
7
|
100
|
|
|
|
20
|
if(ref($refsave->{$_})) { |
|
632
|
3
|
|
|
|
|
9
|
$ref->{$_} = $refsave->{$_}; |
|
633
|
|
|
|
|
|
|
} |
|
634
|
|
|
|
|
|
|
else { |
|
635
|
4
|
|
|
|
|
10
|
$ref->{$_} = [ $refsave->{$_} ]; |
|
636
|
|
|
|
|
|
|
} |
|
637
|
|
|
|
|
|
|
} |
|
638
|
|
|
|
|
|
|
} |
|
639
|
|
|
|
|
|
|
} |
|
640
|
|
|
|
|
|
|
|
|
641
|
|
|
|
|
|
|
|
|
642
|
|
|
|
|
|
|
# Encode the hashref and write to file if necessary |
|
643
|
|
|
|
|
|
|
|
|
644
|
79
|
|
|
|
|
146
|
$self->{_ancestors} = {}; |
|
645
|
79
|
|
|
|
|
227
|
my $xml = $self->value_to_xml($ref, $self->{opt}->{rootname}, ''); |
|
646
|
77
|
|
|
|
|
190
|
delete $self->{_ancestors}; |
|
647
|
|
|
|
|
|
|
|
|
648
|
77
|
100
|
|
|
|
188
|
if($self->{opt}->{xmldecl}) { |
|
649
|
2
|
|
|
|
|
6
|
$xml = $self->{opt}->{xmldecl} . "\n" . $xml; |
|
650
|
|
|
|
|
|
|
} |
|
651
|
|
|
|
|
|
|
|
|
652
|
77
|
100
|
|
|
|
243
|
if($self->{opt}->{outputfile}) { |
|
|
|
100
|
|
|
|
|
|
|
653
|
2
|
100
|
|
|
|
8
|
if(ref($self->{opt}->{outputfile})) { |
|
654
|
1
|
|
|
|
|
3
|
my $fh = $self->{opt}->{outputfile}; |
|
655
|
1
|
50
|
33
|
|
|
17
|
if(UNIVERSAL::isa($fh, 'GLOB') and !UNIVERSAL::can($fh, 'print')) { |
|
656
|
0
|
|
|
|
|
0
|
eval { require IO::Handle; }; |
|
|
0
|
|
|
|
|
0
|
|
|
657
|
0
|
0
|
|
|
|
0
|
croak $@ if $@; |
|
658
|
|
|
|
|
|
|
} |
|
659
|
1
|
|
|
|
|
6
|
return($fh->print($xml)); |
|
660
|
|
|
|
|
|
|
} |
|
661
|
|
|
|
|
|
|
else { |
|
662
|
1
|
50
|
|
|
|
107
|
open(my $out, '>', "$self->{opt}->{outputfile}") || |
|
663
|
|
|
|
|
|
|
croak "open($self->{opt}->{outputfile}): $!"; |
|
664
|
1
|
50
|
|
|
|
9
|
binmode($out, ':utf8') if($] >= 5.008); |
|
665
|
1
|
50
|
|
|
|
14
|
print $out $xml or croak "print: $!"; |
|
666
|
1
|
50
|
|
|
|
56
|
close $out or croak "close: $!"; |
|
667
|
|
|
|
|
|
|
} |
|
668
|
|
|
|
|
|
|
} |
|
669
|
|
|
|
|
|
|
elsif($self->{opt}->{handler}) { |
|
670
|
2
|
|
|
|
|
18
|
require XML::SAX; |
|
671
|
|
|
|
|
|
|
my $sp = XML::SAX::ParserFactory->parser( |
|
672
|
|
|
|
|
|
|
Handler => $self->{opt}->{handler} |
|
673
|
2
|
|
|
|
|
14
|
); |
|
674
|
2
|
|
|
|
|
983
|
return($sp->parse_string($xml)); |
|
675
|
|
|
|
|
|
|
} |
|
676
|
|
|
|
|
|
|
else { |
|
677
|
73
|
|
|
|
|
417
|
return($xml); |
|
678
|
|
|
|
|
|
|
} |
|
679
|
|
|
|
|
|
|
} |
|
680
|
|
|
|
|
|
|
|
|
681
|
|
|
|
|
|
|
|
|
682
|
|
|
|
|
|
|
############################################################################## |
|
683
|
|
|
|
|
|
|
# Method: handle_options() |
|
684
|
|
|
|
|
|
|
# |
|
685
|
|
|
|
|
|
|
# Helper routine for both XMLin() and XMLout(). Both routines handle their |
|
686
|
|
|
|
|
|
|
# first argument and assume all other args are options handled by this routine. |
|
687
|
|
|
|
|
|
|
# Saves a hash of options in $self->{opt}. |
|
688
|
|
|
|
|
|
|
# |
|
689
|
|
|
|
|
|
|
# If default options were passed to the constructor, they will be retrieved |
|
690
|
|
|
|
|
|
|
# here and merged with options supplied to the method call. |
|
691
|
|
|
|
|
|
|
# |
|
692
|
|
|
|
|
|
|
# First argument should be the string 'in' or the string 'out'. |
|
693
|
|
|
|
|
|
|
# |
|
694
|
|
|
|
|
|
|
# Remaining arguments should be name=>value pairs. Sets up default values |
|
695
|
|
|
|
|
|
|
# for options not supplied. Unrecognised options are a fatal error. |
|
696
|
|
|
|
|
|
|
# |
|
697
|
|
|
|
|
|
|
|
|
698
|
|
|
|
|
|
|
sub handle_options { |
|
699
|
271
|
|
|
271
|
1
|
410
|
my $self = shift; |
|
700
|
271
|
|
|
|
|
427
|
my $dirn = shift; |
|
701
|
|
|
|
|
|
|
|
|
702
|
|
|
|
|
|
|
|
|
703
|
|
|
|
|
|
|
# Determine valid options based on context |
|
704
|
|
|
|
|
|
|
|
|
705
|
271
|
|
|
|
|
371
|
my %known_opt; |
|
706
|
271
|
100
|
|
|
|
595
|
if($dirn eq 'in') { |
|
707
|
187
|
|
|
|
|
2117
|
@known_opt{@KnownOptIn} = @KnownOptIn; |
|
708
|
|
|
|
|
|
|
} |
|
709
|
|
|
|
|
|
|
else { |
|
710
|
84
|
|
|
|
|
744
|
@known_opt{@KnownOptOut} = @KnownOptOut; |
|
711
|
|
|
|
|
|
|
} |
|
712
|
|
|
|
|
|
|
|
|
713
|
|
|
|
|
|
|
|
|
714
|
|
|
|
|
|
|
# Store supplied options in hashref and weed out invalid ones |
|
715
|
|
|
|
|
|
|
|
|
716
|
271
|
100
|
|
|
|
1013
|
if(@_ % 2) { |
|
717
|
2
|
|
|
|
|
257
|
croak "Options must be name=>value pairs (odd number supplied)"; |
|
718
|
|
|
|
|
|
|
} |
|
719
|
269
|
|
|
|
|
702
|
my %raw_opt = @_; |
|
720
|
269
|
|
|
|
|
430
|
my $opt = {}; |
|
721
|
269
|
|
|
|
|
643
|
$self->{opt} = $opt; |
|
722
|
|
|
|
|
|
|
|
|
723
|
269
|
|
|
|
|
911
|
while(my($key, $val) = each %raw_opt) { |
|
724
|
269
|
|
|
|
|
473
|
my $lkey = lc($key); |
|
725
|
269
|
|
|
|
|
449
|
$lkey =~ s/_//g; |
|
726
|
269
|
100
|
|
|
|
1006
|
croak "Unrecognised option: $key" unless($known_opt{$lkey}); |
|
727
|
267
|
|
|
|
|
1228
|
$opt->{$lkey} = $val; |
|
728
|
|
|
|
|
|
|
} |
|
729
|
|
|
|
|
|
|
|
|
730
|
|
|
|
|
|
|
|
|
731
|
|
|
|
|
|
|
# Merge in options passed to constructor |
|
732
|
|
|
|
|
|
|
|
|
733
|
267
|
|
|
|
|
1222
|
foreach (keys(%known_opt)) { |
|
734
|
4991
|
100
|
|
|
|
10506
|
unless(exists($opt->{$_})) { |
|
735
|
4724
|
100
|
|
|
|
11666
|
if(exists($self->{def_opt}->{$_})) { |
|
736
|
314
|
|
|
|
|
772
|
$opt->{$_} = $self->{def_opt}->{$_}; |
|
737
|
|
|
|
|
|
|
} |
|
738
|
|
|
|
|
|
|
} |
|
739
|
|
|
|
|
|
|
} |
|
740
|
|
|
|
|
|
|
|
|
741
|
|
|
|
|
|
|
|
|
742
|
|
|
|
|
|
|
# Set sensible defaults if not supplied |
|
743
|
|
|
|
|
|
|
|
|
744
|
267
|
100
|
|
|
|
1056
|
if(exists($opt->{rootname})) { |
|
745
|
20
|
100
|
|
|
|
58
|
unless(defined($opt->{rootname})) { |
|
746
|
1
|
|
|
|
|
3
|
$opt->{rootname} = ''; |
|
747
|
|
|
|
|
|
|
} |
|
748
|
|
|
|
|
|
|
} |
|
749
|
|
|
|
|
|
|
else { |
|
750
|
247
|
|
|
|
|
535
|
$opt->{rootname} = $DefRootName; |
|
751
|
|
|
|
|
|
|
} |
|
752
|
|
|
|
|
|
|
|
|
753
|
267
|
100
|
100
|
|
|
788
|
if($opt->{xmldecl} and $opt->{xmldecl} eq '1') { |
|
754
|
1
|
|
|
|
|
3
|
$opt->{xmldecl} = $DefXmlDecl; |
|
755
|
|
|
|
|
|
|
} |
|
756
|
|
|
|
|
|
|
|
|
757
|
267
|
100
|
|
|
|
599
|
if(exists($opt->{contentkey})) { |
|
758
|
69
|
100
|
|
|
|
301
|
if($opt->{contentkey} =~ m{^-(.*)$}) { |
|
759
|
61
|
|
|
|
|
216
|
$opt->{contentkey} = $1; |
|
760
|
61
|
|
|
|
|
136
|
$opt->{collapseagain} = 1; |
|
761
|
|
|
|
|
|
|
} |
|
762
|
|
|
|
|
|
|
} |
|
763
|
|
|
|
|
|
|
else { |
|
764
|
198
|
|
|
|
|
386
|
$opt->{contentkey} = $DefContentKey; |
|
765
|
|
|
|
|
|
|
} |
|
766
|
|
|
|
|
|
|
|
|
767
|
267
|
100
|
|
|
|
665
|
unless(exists($opt->{normalisespace})) { |
|
768
|
263
|
|
|
|
|
502
|
$opt->{normalisespace} = $opt->{normalizespace}; |
|
769
|
|
|
|
|
|
|
} |
|
770
|
267
|
100
|
|
|
|
754
|
$opt->{normalisespace} = 0 unless(defined($opt->{normalisespace})); |
|
771
|
|
|
|
|
|
|
|
|
772
|
|
|
|
|
|
|
# Cleanups for values assumed to be arrays later |
|
773
|
|
|
|
|
|
|
|
|
774
|
267
|
100
|
|
|
|
556
|
if($opt->{searchpath}) { |
|
775
|
3
|
100
|
|
|
|
10
|
unless(ref($opt->{searchpath})) { |
|
776
|
2
|
|
|
|
|
9
|
$opt->{searchpath} = [ $opt->{searchpath} ]; |
|
777
|
|
|
|
|
|
|
} |
|
778
|
|
|
|
|
|
|
} |
|
779
|
|
|
|
|
|
|
else { |
|
780
|
264
|
|
|
|
|
588
|
$opt->{searchpath} = [ ]; |
|
781
|
|
|
|
|
|
|
} |
|
782
|
|
|
|
|
|
|
|
|
783
|
267
|
100
|
100
|
|
|
882
|
if($opt->{cache} and !ref($opt->{cache})) { |
|
784
|
18
|
|
|
|
|
99
|
$opt->{cache} = [ $opt->{cache} ]; |
|
785
|
|
|
|
|
|
|
} |
|
786
|
267
|
100
|
|
|
|
692
|
if($opt->{cache}) { |
|
787
|
19
|
|
|
|
|
40
|
$_ = lc($_) foreach (@{$opt->{cache}}); |
|
|
19
|
|
|
|
|
117
|
|
|
788
|
19
|
|
|
|
|
47
|
foreach my $scheme (@{$opt->{cache}}) { |
|
|
19
|
|
|
|
|
64
|
|
|
789
|
19
|
|
|
|
|
54
|
my $method = 'cache_read_' . $scheme; |
|
790
|
19
|
100
|
|
|
|
768
|
croak "Unsupported caching scheme: $scheme" |
|
791
|
|
|
|
|
|
|
unless($self->can($method)); |
|
792
|
|
|
|
|
|
|
} |
|
793
|
|
|
|
|
|
|
} |
|
794
|
|
|
|
|
|
|
|
|
795
|
265
|
50
|
|
|
|
567
|
if(exists($opt->{parseropts})) { |
|
796
|
0
|
0
|
|
|
|
0
|
if(warnings::enabled()) { |
|
797
|
0
|
|
|
|
|
0
|
carp "Warning: " . |
|
798
|
|
|
|
|
|
|
"'ParserOpts' is deprecated, contact the author if you need it"; |
|
799
|
|
|
|
|
|
|
} |
|
800
|
|
|
|
|
|
|
} |
|
801
|
|
|
|
|
|
|
else { |
|
802
|
265
|
|
|
|
|
609
|
$opt->{parseropts} = [ ]; |
|
803
|
|
|
|
|
|
|
} |
|
804
|
|
|
|
|
|
|
|
|
805
|
|
|
|
|
|
|
|
|
806
|
|
|
|
|
|
|
# Special cleanup for {forcearray} which could be regex, arrayref or boolean |
|
807
|
|
|
|
|
|
|
# or left to default to 0 |
|
808
|
|
|
|
|
|
|
|
|
809
|
265
|
100
|
|
|
|
581
|
if(exists($opt->{forcearray})) { |
|
810
|
36
|
100
|
|
|
|
128
|
if(ref($opt->{forcearray}) eq 'Regexp') { |
|
811
|
1
|
|
|
|
|
4
|
$opt->{forcearray} = [ $opt->{forcearray} ]; |
|
812
|
|
|
|
|
|
|
} |
|
813
|
|
|
|
|
|
|
|
|
814
|
36
|
100
|
|
|
|
103
|
if(ref($opt->{forcearray}) eq 'ARRAY') { |
|
815
|
16
|
|
|
|
|
24
|
my @force_list = @{$opt->{forcearray}}; |
|
|
16
|
|
|
|
|
47
|
|
|
816
|
16
|
100
|
|
|
|
44
|
if(@force_list) { |
|
817
|
14
|
|
|
|
|
33
|
$opt->{forcearray} = {}; |
|
818
|
14
|
|
|
|
|
32
|
foreach my $tag (@force_list) { |
|
819
|
21
|
100
|
|
|
|
47
|
if(ref($tag) eq 'Regexp') { |
|
820
|
3
|
|
|
|
|
5
|
push @{$opt->{forcearray}->{_regex}}, $tag; |
|
|
3
|
|
|
|
|
13
|
|
|
821
|
|
|
|
|
|
|
} |
|
822
|
|
|
|
|
|
|
else { |
|
823
|
18
|
|
|
|
|
55
|
$opt->{forcearray}->{$tag} = 1; |
|
824
|
|
|
|
|
|
|
} |
|
825
|
|
|
|
|
|
|
} |
|
826
|
|
|
|
|
|
|
} |
|
827
|
|
|
|
|
|
|
else { |
|
828
|
2
|
|
|
|
|
4
|
$opt->{forcearray} = 0; |
|
829
|
|
|
|
|
|
|
} |
|
830
|
|
|
|
|
|
|
} |
|
831
|
|
|
|
|
|
|
else { |
|
832
|
20
|
100
|
|
|
|
63
|
$opt->{forcearray} = ( $opt->{forcearray} ? 1 : 0 ); |
|
833
|
|
|
|
|
|
|
} |
|
834
|
|
|
|
|
|
|
} |
|
835
|
|
|
|
|
|
|
else { |
|
836
|
229
|
100
|
100
|
|
|
654
|
if($opt->{strictmode} and $dirn eq 'in') { |
|
837
|
3
|
|
|
|
|
509
|
croak "No value specified for 'ForceArray' option in call to XML$dirn()"; |
|
838
|
|
|
|
|
|
|
} |
|
839
|
226
|
|
|
|
|
551
|
$opt->{forcearray} = 0; |
|
840
|
|
|
|
|
|
|
} |
|
841
|
|
|
|
|
|
|
|
|
842
|
|
|
|
|
|
|
|
|
843
|
|
|
|
|
|
|
# Special cleanup for {keyattr} which could be arrayref or hashref or left |
|
844
|
|
|
|
|
|
|
# to default to arrayref |
|
845
|
|
|
|
|
|
|
|
|
846
|
262
|
100
|
|
|
|
569
|
if(exists($opt->{keyattr})) { |
|
847
|
78
|
100
|
|
|
|
200
|
if(ref($opt->{keyattr})) { |
|
848
|
74
|
100
|
|
|
|
227
|
if(ref($opt->{keyattr}) eq 'HASH') { |
|
849
|
|
|
|
|
|
|
|
|
850
|
|
|
|
|
|
|
# Make a copy so we can mess with it |
|
851
|
|
|
|
|
|
|
|
|
852
|
58
|
|
|
|
|
74
|
$opt->{keyattr} = { %{$opt->{keyattr}} }; |
|
|
58
|
|
|
|
|
216
|
|
|
853
|
|
|
|
|
|
|
|
|
854
|
|
|
|
|
|
|
|
|
855
|
|
|
|
|
|
|
# Convert keyattr => { elem => '+attr' } |
|
856
|
|
|
|
|
|
|
# to keyattr => { elem => [ 'attr', '+' ] } |
|
857
|
|
|
|
|
|
|
|
|
858
|
58
|
|
|
|
|
107
|
foreach my $el (keys(%{$opt->{keyattr}})) { |
|
|
58
|
|
|
|
|
170
|
|
|
859
|
64
|
50
|
|
|
|
308
|
if($opt->{keyattr}->{$el} =~ /^(\+|-)?(.*)$/) { |
|
860
|
64
|
100
|
|
|
|
316
|
$opt->{keyattr}->{$el} = [ $2, ($1 ? $1 : '') ]; |
|
861
|
64
|
100
|
100
|
|
|
337
|
if($opt->{strictmode} and $dirn eq 'in') { |
|
862
|
9
|
100
|
|
|
|
28
|
next if($opt->{forcearray} == 1); |
|
863
|
|
|
|
|
|
|
next if(ref($opt->{forcearray}) eq 'HASH' |
|
864
|
6
|
100
|
66
|
|
|
28
|
and $opt->{forcearray}->{$el}); |
|
865
|
4
|
|
|
|
|
522
|
croak "<$el> set in KeyAttr but not in ForceArray"; |
|
866
|
|
|
|
|
|
|
} |
|
867
|
|
|
|
|
|
|
} |
|
868
|
|
|
|
|
|
|
else { |
|
869
|
0
|
|
|
|
|
0
|
delete($opt->{keyattr}->{$el}); # Never reached (famous last words?) |
|
870
|
|
|
|
|
|
|
} |
|
871
|
|
|
|
|
|
|
} |
|
872
|
|
|
|
|
|
|
} |
|
873
|
|
|
|
|
|
|
else { |
|
874
|
16
|
100
|
|
|
|
21
|
if(@{$opt->{keyattr}} == 0) { |
|
|
16
|
|
|
|
|
59
|
|
|
875
|
4
|
|
|
|
|
10
|
delete($opt->{keyattr}); |
|
876
|
|
|
|
|
|
|
} |
|
877
|
|
|
|
|
|
|
} |
|
878
|
|
|
|
|
|
|
} |
|
879
|
|
|
|
|
|
|
else { |
|
880
|
4
|
|
|
|
|
17
|
$opt->{keyattr} = [ $opt->{keyattr} ]; |
|
881
|
|
|
|
|
|
|
} |
|
882
|
|
|
|
|
|
|
} |
|
883
|
|
|
|
|
|
|
else { |
|
884
|
184
|
100
|
|
|
|
436
|
if($opt->{strictmode}) { |
|
885
|
4
|
|
|
|
|
528
|
croak "No value specified for 'KeyAttr' option in call to XML$dirn()"; |
|
886
|
|
|
|
|
|
|
} |
|
887
|
180
|
|
|
|
|
603
|
$opt->{keyattr} = [ @DefKeyAttr ]; |
|
888
|
|
|
|
|
|
|
} |
|
889
|
|
|
|
|
|
|
|
|
890
|
|
|
|
|
|
|
|
|
891
|
|
|
|
|
|
|
# Special cleanup for {valueattr} which could be arrayref or hashref |
|
892
|
|
|
|
|
|
|
|
|
893
|
254
|
100
|
|
|
|
658
|
if(exists($opt->{valueattr})) { |
|
894
|
5
|
100
|
|
|
|
22
|
if(ref($opt->{valueattr}) eq 'ARRAY') { |
|
895
|
2
|
|
|
|
|
4
|
$opt->{valueattrlist} = {}; |
|
896
|
2
|
|
|
|
|
4
|
$opt->{valueattrlist}->{$_} = 1 foreach(@{ delete $opt->{valueattr} }); |
|
|
2
|
|
|
|
|
10
|
|
|
897
|
|
|
|
|
|
|
} |
|
898
|
|
|
|
|
|
|
} |
|
899
|
|
|
|
|
|
|
|
|
900
|
|
|
|
|
|
|
# make sure there's nothing weird in {grouptags} |
|
901
|
|
|
|
|
|
|
|
|
902
|
254
|
100
|
|
|
|
581
|
if($opt->{grouptags}) { |
|
903
|
|
|
|
|
|
|
croak "Illegal value for 'GroupTags' option - expected a hashref" |
|
904
|
14
|
100
|
|
|
|
277
|
unless UNIVERSAL::isa($opt->{grouptags}, 'HASH'); |
|
905
|
|
|
|
|
|
|
|
|
906
|
13
|
|
|
|
|
22
|
while(my($key, $val) = each %{$opt->{grouptags}}) { |
|
|
27
|
|
|
|
|
99
|
|
|
907
|
15
|
100
|
|
|
|
46
|
next if $key ne $val; |
|
908
|
1
|
|
|
|
|
148
|
croak "Bad value in GroupTags: '$key' => '$val'"; |
|
909
|
|
|
|
|
|
|
} |
|
910
|
|
|
|
|
|
|
} |
|
911
|
|
|
|
|
|
|
|
|
912
|
|
|
|
|
|
|
|
|
913
|
|
|
|
|
|
|
# Check the {variables} option is valid and initialise variables hash |
|
914
|
|
|
|
|
|
|
|
|
915
|
252
|
100
|
100
|
|
|
714
|
if($opt->{variables} and !UNIVERSAL::isa($opt->{variables}, 'HASH')) { |
|
916
|
1
|
|
|
|
|
153
|
croak "Illegal value for 'Variables' option - expected a hashref"; |
|
917
|
|
|
|
|
|
|
} |
|
918
|
|
|
|
|
|
|
|
|
919
|
251
|
100
|
|
|
|
1721
|
if($opt->{variables}) { |
|
|
|
100
|
|
|
|
|
|
|
920
|
4
|
|
|
|
|
8
|
$self->{_var_values} = { %{$opt->{variables}} }; |
|
|
4
|
|
|
|
|
31
|
|
|
921
|
|
|
|
|
|
|
} |
|
922
|
|
|
|
|
|
|
elsif($opt->{varattr}) { |
|
923
|
2
|
|
|
|
|
12
|
$self->{_var_values} = {}; |
|
924
|
|
|
|
|
|
|
} |
|
925
|
|
|
|
|
|
|
|
|
926
|
|
|
|
|
|
|
} |
|
927
|
|
|
|
|
|
|
|
|
928
|
|
|
|
|
|
|
|
|
929
|
|
|
|
|
|
|
############################################################################## |
|
930
|
|
|
|
|
|
|
# Method: find_xml_file() |
|
931
|
|
|
|
|
|
|
# |
|
932
|
|
|
|
|
|
|
# Helper routine for XMLin(). |
|
933
|
|
|
|
|
|
|
# Takes a filename, and a list of directories, attempts to locate the file in |
|
934
|
|
|
|
|
|
|
# the directories listed. |
|
935
|
|
|
|
|
|
|
# Returns a full pathname on success; croaks on failure. |
|
936
|
|
|
|
|
|
|
# |
|
937
|
|
|
|
|
|
|
|
|
938
|
|
|
|
|
|
|
sub find_xml_file { |
|
939
|
27
|
|
|
27
|
0
|
57
|
my $self = shift; |
|
940
|
27
|
|
|
|
|
55
|
my $file = shift; |
|
941
|
27
|
|
|
|
|
72
|
my @search_path = @_; |
|
942
|
|
|
|
|
|
|
|
|
943
|
|
|
|
|
|
|
|
|
944
|
27
|
|
|
|
|
298
|
require File::Basename; |
|
945
|
27
|
|
|
|
|
139
|
require File::Spec; |
|
946
|
|
|
|
|
|
|
|
|
947
|
27
|
|
|
|
|
941
|
my($filename, $filedir) = File::Basename::fileparse($file); |
|
948
|
|
|
|
|
|
|
|
|
949
|
27
|
100
|
|
|
|
122
|
if($filename ne $file) { # Ignore searchpath if dir component |
|
950
|
23
|
100
|
|
|
|
829
|
return($file) if(-e $file); |
|
951
|
|
|
|
|
|
|
} |
|
952
|
|
|
|
|
|
|
else { |
|
953
|
4
|
|
|
|
|
5
|
my($path); |
|
954
|
4
|
|
|
|
|
9
|
foreach $path (@search_path) { |
|
955
|
5
|
|
|
|
|
49
|
my $fullpath = File::Spec->catfile($path, $file); |
|
956
|
5
|
100
|
|
|
|
104
|
return($fullpath) if(-e $fullpath); |
|
957
|
|
|
|
|
|
|
} |
|
958
|
|
|
|
|
|
|
} |
|
959
|
|
|
|
|
|
|
|
|
960
|
|
|
|
|
|
|
# If user did not supply a search path, default to current directory |
|
961
|
|
|
|
|
|
|
|
|
962
|
3
|
100
|
|
|
|
11
|
if(!@search_path) { |
|
963
|
1
|
50
|
|
|
|
19
|
return($file) if(-e $file); |
|
964
|
1
|
|
|
|
|
140
|
croak "File does not exist: $file"; |
|
965
|
|
|
|
|
|
|
} |
|
966
|
|
|
|
|
|
|
|
|
967
|
2
|
|
|
|
|
571
|
croak "Could not find $file in ", join(':', @search_path); |
|
968
|
|
|
|
|
|
|
} |
|
969
|
|
|
|
|
|
|
|
|
970
|
|
|
|
|
|
|
|
|
971
|
|
|
|
|
|
|
############################################################################## |
|
972
|
|
|
|
|
|
|
# Method: collapse() |
|
973
|
|
|
|
|
|
|
# |
|
974
|
|
|
|
|
|
|
# Helper routine for XMLin(). This routine really comprises the 'smarts' (or |
|
975
|
|
|
|
|
|
|
# value add) of this module. |
|
976
|
|
|
|
|
|
|
# |
|
977
|
|
|
|
|
|
|
# Takes the parse tree that XML::Parser produced from the supplied XML and |
|
978
|
|
|
|
|
|
|
# recurses through it 'collapsing' unnecessary levels of indirection (nested |
|
979
|
|
|
|
|
|
|
# arrays etc) to produce a data structure that is easier to work with. |
|
980
|
|
|
|
|
|
|
# |
|
981
|
|
|
|
|
|
|
# Elements in the original parser tree are represented as an element name |
|
982
|
|
|
|
|
|
|
# followed by an arrayref. The first element of the array is a hashref |
|
983
|
|
|
|
|
|
|
# containing the attributes. The rest of the array contains a list of any |
|
984
|
|
|
|
|
|
|
# nested elements as name+arrayref pairs: |
|
985
|
|
|
|
|
|
|
# |
|
986
|
|
|
|
|
|
|
# , [ { }, , [ ... ], ... ] |
|
987
|
|
|
|
|
|
|
# |
|
988
|
|
|
|
|
|
|
# The special element name '0' (zero) flags text content. |
|
989
|
|
|
|
|
|
|
# |
|
990
|
|
|
|
|
|
|
# This routine cuts down the noise by discarding any text content consisting of |
|
991
|
|
|
|
|
|
|
# only whitespace and then moves the nested elements into the attribute hash |
|
992
|
|
|
|
|
|
|
# using the name of the nested element as the hash key and the collapsed |
|
993
|
|
|
|
|
|
|
# version of the nested element as the value. Multiple nested elements with |
|
994
|
|
|
|
|
|
|
# the same name will initially be represented as an arrayref, but this may be |
|
995
|
|
|
|
|
|
|
# 'folded' into a hashref depending on the value of the keyattr option. |
|
996
|
|
|
|
|
|
|
# |
|
997
|
|
|
|
|
|
|
|
|
998
|
|
|
|
|
|
|
sub collapse { |
|
999
|
3537
|
|
|
3537
|
0
|
4682
|
my $self = shift; |
|
1000
|
|
|
|
|
|
|
|
|
1001
|
|
|
|
|
|
|
|
|
1002
|
|
|
|
|
|
|
# Start with the hash of attributes |
|
1003
|
|
|
|
|
|
|
|
|
1004
|
3537
|
|
|
|
|
4908
|
my $attr = shift; |
|
1005
|
3537
|
100
|
|
|
|
11570
|
if($self->{opt}->{noattr}) { # Discard if 'noattr' set |
|
|
|
100
|
|
|
|
|
|
|
1006
|
36
|
|
|
|
|
83
|
$attr = $self->new_hashref; |
|
1007
|
|
|
|
|
|
|
} |
|
1008
|
|
|
|
|
|
|
elsif($self->{opt}->{normalisespace} == 2) { |
|
1009
|
15
|
|
|
|
|
54
|
while(my($key, $value) = each %$attr) { |
|
1010
|
2
|
|
|
|
|
6
|
$attr->{$key} = $self->normalise_space($value) |
|
1011
|
|
|
|
|
|
|
} |
|
1012
|
|
|
|
|
|
|
} |
|
1013
|
|
|
|
|
|
|
|
|
1014
|
|
|
|
|
|
|
|
|
1015
|
|
|
|
|
|
|
# Do variable substitutions |
|
1016
|
|
|
|
|
|
|
|
|
1017
|
3537
|
100
|
|
|
|
8199
|
if(my $var = $self->{_var_values}) { |
|
1018
|
37
|
|
|
|
|
139
|
while(my($key, $val) = each(%$attr)) { |
|
1019
|
30
|
|
|
|
|
76
|
$val =~ s^\$\{([\w.]+)\}^ $self->get_var($1) ^ge; |
|
|
4
|
|
|
|
|
9
|
|
|
1020
|
30
|
|
|
|
|
137
|
$attr->{$key} = $val; |
|
1021
|
|
|
|
|
|
|
} |
|
1022
|
|
|
|
|
|
|
} |
|
1023
|
|
|
|
|
|
|
|
|
1024
|
|
|
|
|
|
|
|
|
1025
|
|
|
|
|
|
|
# Roll up 'value' attributes (but only if no nested elements) |
|
1026
|
|
|
|
|
|
|
|
|
1027
|
3537
|
100
|
100
|
|
|
13725
|
if(!@_ and keys %$attr == 1) { |
|
1028
|
42
|
|
|
|
|
119
|
my($k) = keys %$attr; |
|
1029
|
42
|
100
|
66
|
|
|
151
|
if($self->{opt}->{valueattrlist} and $self->{opt}->{valueattrlist}->{$k}) { |
|
1030
|
7
|
|
|
|
|
22
|
return $attr->{$k}; |
|
1031
|
|
|
|
|
|
|
} |
|
1032
|
|
|
|
|
|
|
} |
|
1033
|
|
|
|
|
|
|
|
|
1034
|
|
|
|
|
|
|
|
|
1035
|
|
|
|
|
|
|
# Add any nested elements |
|
1036
|
|
|
|
|
|
|
|
|
1037
|
3530
|
|
|
|
|
4192
|
my($key, $val); |
|
1038
|
3530
|
|
|
|
|
7763
|
while(@_) { |
|
1039
|
8308
|
|
|
|
|
11388
|
$key = shift; |
|
1040
|
8308
|
|
|
|
|
12524
|
$val = shift; |
|
1041
|
8308
|
50
|
|
|
|
16167
|
$val = '' if not defined $val; |
|
1042
|
|
|
|
|
|
|
|
|
1043
|
8308
|
100
|
|
|
|
21378
|
if(ref($val)) { |
|
|
|
50
|
|
|
|
|
|
|
1044
|
3376
|
|
|
|
|
8785
|
$val = $self->collapse(@$val); |
|
1045
|
3376
|
100
|
66
|
|
|
8669
|
next if(!defined($val) and $self->{opt}->{suppressempty}); |
|
1046
|
|
|
|
|
|
|
} |
|
1047
|
|
|
|
|
|
|
elsif($key eq '0') { |
|
1048
|
4932
|
100
|
|
|
|
18588
|
next if($val =~ m{^\s*$}s); # Skip all whitespace content |
|
1049
|
|
|
|
|
|
|
|
|
1050
|
|
|
|
|
|
|
$val = $self->normalise_space($val) |
|
1051
|
1268
|
100
|
|
|
|
2775
|
if($self->{opt}->{normalisespace} == 2); |
|
1052
|
|
|
|
|
|
|
|
|
1053
|
|
|
|
|
|
|
# do variable substitutions |
|
1054
|
|
|
|
|
|
|
|
|
1055
|
1268
|
100
|
|
|
|
2670
|
if(my $var = $self->{_var_values}) { |
|
1056
|
26
|
|
|
|
|
104
|
$val =~ s^\$\{(\w+)\}^ $self->get_var($1) ^ge; |
|
|
17
|
|
|
|
|
45
|
|
|
1057
|
|
|
|
|
|
|
} |
|
1058
|
|
|
|
|
|
|
|
|
1059
|
|
|
|
|
|
|
|
|
1060
|
|
|
|
|
|
|
# look for variable definitions |
|
1061
|
|
|
|
|
|
|
|
|
1062
|
1268
|
100
|
|
|
|
2813
|
if(my $var = $self->{opt}->{varattr}) { |
|
1063
|
23
|
100
|
|
|
|
62
|
if(exists $attr->{$var}) { |
|
1064
|
10
|
|
|
|
|
29
|
$self->set_var($attr->{$var}, $val); |
|
1065
|
|
|
|
|
|
|
} |
|
1066
|
|
|
|
|
|
|
} |
|
1067
|
|
|
|
|
|
|
|
|
1068
|
|
|
|
|
|
|
|
|
1069
|
|
|
|
|
|
|
# Collapse text content in element with no attributes to a string |
|
1070
|
|
|
|
|
|
|
|
|
1071
|
1268
|
100
|
66
|
|
|
4506
|
if(!%$attr and !@_) { |
|
1072
|
|
|
|
|
|
|
return($self->{opt}->{forcecontent} ? |
|
1073
|
1141
|
100
|
|
|
|
3873
|
{ $self->{opt}->{contentkey} => $val } : $val |
|
1074
|
|
|
|
|
|
|
); |
|
1075
|
|
|
|
|
|
|
} |
|
1076
|
127
|
|
|
|
|
285
|
$key = $self->{opt}->{contentkey}; |
|
1077
|
|
|
|
|
|
|
} |
|
1078
|
|
|
|
|
|
|
|
|
1079
|
|
|
|
|
|
|
|
|
1080
|
|
|
|
|
|
|
# Combine duplicate attributes into arrayref if required |
|
1081
|
|
|
|
|
|
|
|
|
1082
|
3497
|
100
|
100
|
|
|
11430
|
if(exists($attr->{$key})) { |
|
|
|
100
|
|
|
|
|
|
|
1083
|
2859
|
100
|
|
|
|
9167
|
if(UNIVERSAL::isa($attr->{$key}, 'ARRAY')) { |
|
1084
|
2729
|
|
|
|
|
3204
|
push(@{$attr->{$key}}, $val); |
|
|
2729
|
|
|
|
|
10281
|
|
|
1085
|
|
|
|
|
|
|
} |
|
1086
|
|
|
|
|
|
|
else { |
|
1087
|
130
|
|
|
|
|
553
|
$attr->{$key} = [ $attr->{$key}, $val ]; |
|
1088
|
|
|
|
|
|
|
} |
|
1089
|
|
|
|
|
|
|
} |
|
1090
|
|
|
|
|
|
|
elsif(defined($val) and UNIVERSAL::isa($val, 'ARRAY')) { |
|
1091
|
4
|
|
|
|
|
19
|
$attr->{$key} = [ $val ]; |
|
1092
|
|
|
|
|
|
|
} |
|
1093
|
|
|
|
|
|
|
else { |
|
1094
|
634
|
100
|
66
|
|
|
3623
|
if( $key ne $self->{opt}->{contentkey} |
|
|
|
|
66
|
|
|
|
|
|
1095
|
|
|
|
|
|
|
and ( |
|
1096
|
|
|
|
|
|
|
($self->{opt}->{forcearray} == 1) |
|
1097
|
|
|
|
|
|
|
or ( |
|
1098
|
|
|
|
|
|
|
(ref($self->{opt}->{forcearray}) eq 'HASH') |
|
1099
|
|
|
|
|
|
|
and ( |
|
1100
|
|
|
|
|
|
|
$self->{opt}->{forcearray}->{$key} |
|
1101
|
|
|
|
|
|
|
or (grep $key =~ $_, @{$self->{opt}->{forcearray}->{_regex}}) |
|
1102
|
|
|
|
|
|
|
) |
|
1103
|
|
|
|
|
|
|
) |
|
1104
|
|
|
|
|
|
|
) |
|
1105
|
|
|
|
|
|
|
) { |
|
1106
|
194
|
|
|
|
|
803
|
$attr->{$key} = [ $val ]; |
|
1107
|
|
|
|
|
|
|
} |
|
1108
|
|
|
|
|
|
|
else { |
|
1109
|
440
|
|
|
|
|
1728
|
$attr->{$key} = $val; |
|
1110
|
|
|
|
|
|
|
} |
|
1111
|
|
|
|
|
|
|
} |
|
1112
|
|
|
|
|
|
|
|
|
1113
|
|
|
|
|
|
|
} |
|
1114
|
|
|
|
|
|
|
|
|
1115
|
|
|
|
|
|
|
|
|
1116
|
|
|
|
|
|
|
# Turn arrayrefs into hashrefs if key fields present |
|
1117
|
|
|
|
|
|
|
|
|
1118
|
2389
|
100
|
|
|
|
5918
|
if($self->{opt}->{keyattr}) { |
|
1119
|
2367
|
|
|
|
|
9608
|
while(($key,$val) = each %$attr) { |
|
1120
|
4725
|
100
|
100
|
|
|
39181
|
if(defined($val) and UNIVERSAL::isa($val, 'ARRAY')) { |
|
1121
|
322
|
|
|
|
|
791
|
$attr->{$key} = $self->array_to_hash($key, $val); |
|
1122
|
|
|
|
|
|
|
} |
|
1123
|
|
|
|
|
|
|
} |
|
1124
|
|
|
|
|
|
|
} |
|
1125
|
|
|
|
|
|
|
|
|
1126
|
|
|
|
|
|
|
|
|
1127
|
|
|
|
|
|
|
# disintermediate grouped tags |
|
1128
|
|
|
|
|
|
|
|
|
1129
|
2384
|
100
|
|
|
|
5628
|
if($self->{opt}->{grouptags}) { |
|
1130
|
26
|
|
|
|
|
103
|
while(my($key, $val) = each(%$attr)) { |
|
1131
|
51
|
100
|
100
|
|
|
321
|
next unless(UNIVERSAL::isa($val, 'HASH') and (keys %$val == 1)); |
|
1132
|
10
|
100
|
|
|
|
36
|
next unless(exists($self->{opt}->{grouptags}->{$key})); |
|
1133
|
|
|
|
|
|
|
|
|
1134
|
9
|
|
|
|
|
24
|
my($child_key, $child_val) = %$val; |
|
1135
|
|
|
|
|
|
|
|
|
1136
|
9
|
100
|
|
|
|
32
|
if($self->{opt}->{grouptags}->{$key} eq $child_key) { |
|
1137
|
8
|
|
|
|
|
36
|
$attr->{$key}= $child_val; |
|
1138
|
|
|
|
|
|
|
} |
|
1139
|
|
|
|
|
|
|
} |
|
1140
|
|
|
|
|
|
|
} |
|
1141
|
|
|
|
|
|
|
|
|
1142
|
|
|
|
|
|
|
|
|
1143
|
|
|
|
|
|
|
# Fold hashes containing a single anonymous array up into just the array |
|
1144
|
|
|
|
|
|
|
|
|
1145
|
2384
|
|
|
|
|
3599
|
my $count = scalar keys %$attr; |
|
1146
|
2384
|
100
|
66
|
|
|
5655
|
if($count == 1 |
|
|
|
|
66
|
|
|
|
|
|
1147
|
|
|
|
|
|
|
and exists $attr->{anon} |
|
1148
|
|
|
|
|
|
|
and UNIVERSAL::isa($attr->{anon}, 'ARRAY') |
|
1149
|
|
|
|
|
|
|
) { |
|
1150
|
61
|
|
|
|
|
199
|
return($attr->{anon}); |
|
1151
|
|
|
|
|
|
|
} |
|
1152
|
|
|
|
|
|
|
|
|
1153
|
|
|
|
|
|
|
|
|
1154
|
|
|
|
|
|
|
# Do the right thing if hash is empty, otherwise just return it |
|
1155
|
|
|
|
|
|
|
|
|
1156
|
2323
|
100
|
66
|
|
|
5001
|
if(!%$attr and exists($self->{opt}->{suppressempty})) { |
|
1157
|
11
|
100
|
100
|
|
|
51
|
if(defined($self->{opt}->{suppressempty}) and |
|
1158
|
|
|
|
|
|
|
$self->{opt}->{suppressempty} eq '') { |
|
1159
|
2
|
|
|
|
|
6
|
return(''); |
|
1160
|
|
|
|
|
|
|
} |
|
1161
|
9
|
|
|
|
|
27
|
return(undef); |
|
1162
|
|
|
|
|
|
|
} |
|
1163
|
|
|
|
|
|
|
|
|
1164
|
|
|
|
|
|
|
|
|
1165
|
|
|
|
|
|
|
# Roll up named elements with named nested 'value' attributes |
|
1166
|
|
|
|
|
|
|
|
|
1167
|
2312
|
100
|
|
|
|
5175
|
if($self->{opt}->{valueattr}) { |
|
1168
|
10
|
|
|
|
|
35
|
while(my($key, $val) = each(%$attr)) { |
|
1169
|
18
|
100
|
|
|
|
82
|
next unless($self->{opt}->{valueattr}->{$key}); |
|
1170
|
4
|
50
|
33
|
|
|
28
|
next unless(UNIVERSAL::isa($val, 'HASH') and (keys %$val == 1)); |
|
1171
|
4
|
|
|
|
|
10
|
my($k) = keys %$val; |
|
1172
|
4
|
50
|
|
|
|
16
|
next unless($k eq $self->{opt}->{valueattr}->{$key}); |
|
1173
|
4
|
|
|
|
|
19
|
$attr->{$key} = $val->{$k}; |
|
1174
|
|
|
|
|
|
|
} |
|
1175
|
|
|
|
|
|
|
} |
|
1176
|
|
|
|
|
|
|
|
|
1177
|
2312
|
|
|
|
|
5526
|
return($attr) |
|
1178
|
|
|
|
|
|
|
|
|
1179
|
|
|
|
|
|
|
} |
|
1180
|
|
|
|
|
|
|
|
|
1181
|
|
|
|
|
|
|
|
|
1182
|
|
|
|
|
|
|
############################################################################## |
|
1183
|
|
|
|
|
|
|
# Method: set_var() |
|
1184
|
|
|
|
|
|
|
# |
|
1185
|
|
|
|
|
|
|
# Called when a variable definition is encountered in the XML. (A variable |
|
1186
|
|
|
|
|
|
|
# definition looks like value where attrname |
|
1187
|
|
|
|
|
|
|
# matches the varattr setting). |
|
1188
|
|
|
|
|
|
|
# |
|
1189
|
|
|
|
|
|
|
|
|
1190
|
|
|
|
|
|
|
sub set_var { |
|
1191
|
10
|
|
|
10
|
0
|
21
|
my($self, $name, $value) = @_; |
|
1192
|
|
|
|
|
|
|
|
|
1193
|
10
|
|
|
|
|
38
|
$self->{_var_values}->{$name} = $value; |
|
1194
|
|
|
|
|
|
|
} |
|
1195
|
|
|
|
|
|
|
|
|
1196
|
|
|
|
|
|
|
|
|
1197
|
|
|
|
|
|
|
############################################################################## |
|
1198
|
|
|
|
|
|
|
# Method: get_var() |
|
1199
|
|
|
|
|
|
|
# |
|
1200
|
|
|
|
|
|
|
# Called during variable substitution to get the value for the named variable. |
|
1201
|
|
|
|
|
|
|
# |
|
1202
|
|
|
|
|
|
|
|
|
1203
|
|
|
|
|
|
|
sub get_var { |
|
1204
|
21
|
|
|
21
|
0
|
46
|
my($self, $name) = @_; |
|
1205
|
|
|
|
|
|
|
|
|
1206
|
21
|
|
|
|
|
45
|
my $value = $self->{_var_values}->{$name}; |
|
1207
|
21
|
100
|
|
|
|
95
|
return $value if(defined($value)); |
|
1208
|
|
|
|
|
|
|
|
|
1209
|
1
|
|
|
|
|
5
|
return '${' . $name . '}'; |
|
1210
|
|
|
|
|
|
|
} |
|
1211
|
|
|
|
|
|
|
|
|
1212
|
|
|
|
|
|
|
|
|
1213
|
|
|
|
|
|
|
############################################################################## |
|
1214
|
|
|
|
|
|
|
# Method: normalise_space() |
|
1215
|
|
|
|
|
|
|
# |
|
1216
|
|
|
|
|
|
|
# Strips leading and trailing whitespace and collapses sequences of whitespace |
|
1217
|
|
|
|
|
|
|
# characters to a single space. |
|
1218
|
|
|
|
|
|
|
# |
|
1219
|
|
|
|
|
|
|
|
|
1220
|
|
|
|
|
|
|
sub normalise_space { |
|
1221
|
16
|
|
|
16
|
0
|
28
|
my($self, $text) = @_; |
|
1222
|
|
|
|
|
|
|
|
|
1223
|
16
|
|
|
|
|
63
|
$text =~ s/^\s+//s; |
|
1224
|
16
|
|
|
|
|
72
|
$text =~ s/\s+$//s; |
|
1225
|
16
|
|
|
|
|
59
|
$text =~ s/\s\s+/ /sg; |
|
1226
|
|
|
|
|
|
|
|
|
1227
|
16
|
|
|
|
|
42
|
return $text; |
|
1228
|
|
|
|
|
|
|
} |
|
1229
|
|
|
|
|
|
|
|
|
1230
|
|
|
|
|
|
|
|
|
1231
|
|
|
|
|
|
|
############################################################################## |
|
1232
|
|
|
|
|
|
|
# Method: array_to_hash() |
|
1233
|
|
|
|
|
|
|
# |
|
1234
|
|
|
|
|
|
|
# Helper routine for collapse(). |
|
1235
|
|
|
|
|
|
|
# Attempts to 'fold' an array of hashes into an hash of hashes. Returns a |
|
1236
|
|
|
|
|
|
|
# reference to the hash on success or the original array if folding is |
|
1237
|
|
|
|
|
|
|
# not possible. Behaviour is controlled by 'keyattr' option. |
|
1238
|
|
|
|
|
|
|
# |
|
1239
|
|
|
|
|
|
|
|
|
1240
|
|
|
|
|
|
|
sub array_to_hash { |
|
1241
|
322
|
|
|
322
|
0
|
489
|
my $self = shift; |
|
1242
|
322
|
|
|
|
|
467
|
my $name = shift; |
|
1243
|
322
|
|
|
|
|
427
|
my $arrayref = shift; |
|
1244
|
|
|
|
|
|
|
|
|
1245
|
322
|
|
|
|
|
722
|
my $hashref = $self->new_hashref; |
|
1246
|
|
|
|
|
|
|
|
|
1247
|
322
|
|
|
|
|
458
|
my($i, $key, $val, $flag); |
|
1248
|
|
|
|
|
|
|
|
|
1249
|
|
|
|
|
|
|
|
|
1250
|
|
|
|
|
|
|
# Handle keyattr => { .... } |
|
1251
|
|
|
|
|
|
|
|
|
1252
|
322
|
100
|
|
|
|
886
|
if(ref($self->{opt}->{keyattr}) eq 'HASH') { |
|
1253
|
189
|
100
|
|
|
|
843
|
return($arrayref) unless(exists($self->{opt}->{keyattr}->{$name})); |
|
1254
|
121
|
|
|
|
|
153
|
($key, $flag) = @{$self->{opt}->{keyattr}->{$name}}; |
|
|
121
|
|
|
|
|
424
|
|
|
1255
|
121
|
|
|
|
|
363
|
for($i = 0; $i < @$arrayref; $i++) { |
|
1256
|
1912
|
100
|
33
|
|
|
8006
|
if(UNIVERSAL::isa($arrayref->[$i], 'HASH') and |
|
1257
|
|
|
|
|
|
|
exists($arrayref->[$i]->{$key}) |
|
1258
|
|
|
|
|
|
|
) { |
|
1259
|
1908
|
|
|
|
|
3081
|
$val = $arrayref->[$i]->{$key}; |
|
1260
|
1908
|
100
|
|
|
|
4116
|
if(ref($val)) { |
|
1261
|
4
|
|
|
|
|
36
|
$self->die_or_warn("<$name> element has non-scalar '$key' key attribute"); |
|
1262
|
2
|
|
|
|
|
62
|
return($arrayref); |
|
1263
|
|
|
|
|
|
|
} |
|
1264
|
|
|
|
|
|
|
$val = $self->normalise_space($val) |
|
1265
|
1904
|
100
|
|
|
|
4424
|
if($self->{opt}->{normalisespace} == 1); |
|
1266
|
|
|
|
|
|
|
$self->die_or_warn("<$name> element has non-unique value in '$key' key attribute: $val") |
|
1267
|
1904
|
100
|
|
|
|
4799
|
if(exists($hashref->{$val})); |
|
1268
|
1903
|
|
|
|
|
2538
|
$hashref->{$val} = $self->new_hashref( %{$arrayref->[$i]} ); |
|
|
1903
|
|
|
|
|
6271
|
|
|
1269
|
1903
|
100
|
|
|
|
4855
|
$hashref->{$val}->{"-$key"} = $hashref->{$val}->{$key} if($flag eq '-'); |
|
1270
|
1903
|
100
|
|
|
|
7978
|
delete $hashref->{$val}->{$key} unless($flag eq '+'); |
|
1271
|
|
|
|
|
|
|
} |
|
1272
|
|
|
|
|
|
|
else { |
|
1273
|
4
|
|
|
|
|
24
|
$self->die_or_warn("<$name> element has no '$key' key attribute"); |
|
1274
|
2
|
|
|
|
|
58
|
return($arrayref); |
|
1275
|
|
|
|
|
|
|
} |
|
1276
|
|
|
|
|
|
|
} |
|
1277
|
|
|
|
|
|
|
} |
|
1278
|
|
|
|
|
|
|
|
|
1279
|
|
|
|
|
|
|
|
|
1280
|
|
|
|
|
|
|
# Or assume keyattr => [ .... ] |
|
1281
|
|
|
|
|
|
|
|
|
1282
|
|
|
|
|
|
|
else { |
|
1283
|
|
|
|
|
|
|
my $default_keys = |
|
1284
|
133
|
|
|
|
|
296
|
join(',', @DefKeyAttr) eq join(',', @{$self->{opt}->{keyattr}}); |
|
|
133
|
|
|
|
|
362
|
|
|
1285
|
|
|
|
|
|
|
|
|
1286
|
133
|
|
|
|
|
359
|
ELEMENT: for($i = 0; $i < @$arrayref; $i++) { |
|
1287
|
205
|
100
|
|
|
|
1142
|
return($arrayref) unless(UNIVERSAL::isa($arrayref->[$i], 'HASH')); |
|
1288
|
|
|
|
|
|
|
|
|
1289
|
129
|
|
|
|
|
164
|
foreach $key (@{$self->{opt}->{keyattr}}) { |
|
|
129
|
|
|
|
|
343
|
|
|
1290
|
173
|
100
|
|
|
|
462
|
if(defined($arrayref->[$i]->{$key})) { |
|
1291
|
119
|
|
|
|
|
208
|
$val = $arrayref->[$i]->{$key}; |
|
1292
|
119
|
100
|
|
|
|
251
|
if(ref($val)) { |
|
1293
|
2
|
100
|
|
|
|
10
|
$self->die_or_warn("<$name> element has non-scalar '$key' key attribute") |
|
1294
|
|
|
|
|
|
|
if not $default_keys; |
|
1295
|
2
|
|
|
|
|
55
|
return($arrayref); |
|
1296
|
|
|
|
|
|
|
} |
|
1297
|
|
|
|
|
|
|
$val = $self->normalise_space($val) |
|
1298
|
117
|
100
|
|
|
|
292
|
if($self->{opt}->{normalisespace} == 1); |
|
1299
|
|
|
|
|
|
|
$self->die_or_warn("<$name> element has non-unique value in '$key' key attribute: $val") |
|
1300
|
117
|
100
|
|
|
|
286
|
if(exists($hashref->{$val})); |
|
1301
|
117
|
|
|
|
|
194
|
$hashref->{$val} = $self->new_hashref( %{$arrayref->[$i]} ); |
|
|
117
|
|
|
|
|
444
|
|
|
1302
|
117
|
|
|
|
|
303
|
delete $hashref->{$val}->{$key}; |
|
1303
|
117
|
|
|
|
|
434
|
next ELEMENT; |
|
1304
|
|
|
|
|
|
|
} |
|
1305
|
|
|
|
|
|
|
} |
|
1306
|
|
|
|
|
|
|
|
|
1307
|
10
|
|
|
|
|
55
|
return($arrayref); # No keyfield matched |
|
1308
|
|
|
|
|
|
|
} |
|
1309
|
|
|
|
|
|
|
} |
|
1310
|
|
|
|
|
|
|
|
|
1311
|
|
|
|
|
|
|
# collapse any hashes which now only have a 'content' key |
|
1312
|
|
|
|
|
|
|
|
|
1313
|
157
|
100
|
|
|
|
448
|
if($self->{opt}->{collapseagain}) { |
|
1314
|
36
|
|
|
|
|
102
|
$hashref = $self->collapse_content($hashref); |
|
1315
|
|
|
|
|
|
|
} |
|
1316
|
|
|
|
|
|
|
|
|
1317
|
157
|
|
|
|
|
791
|
return($hashref); |
|
1318
|
|
|
|
|
|
|
} |
|
1319
|
|
|
|
|
|
|
|
|
1320
|
|
|
|
|
|
|
|
|
1321
|
|
|
|
|
|
|
############################################################################## |
|
1322
|
|
|
|
|
|
|
# Method: die_or_warn() |
|
1323
|
|
|
|
|
|
|
# |
|
1324
|
|
|
|
|
|
|
# Takes a diagnostic message and does one of three things: |
|
1325
|
|
|
|
|
|
|
# 1. dies if strict mode is enabled |
|
1326
|
|
|
|
|
|
|
# 2. warns if warnings are enabled but strict mode is not |
|
1327
|
|
|
|
|
|
|
# 3. ignores message and returns silently if neither strict mode nor warnings |
|
1328
|
|
|
|
|
|
|
# are enabled |
|
1329
|
|
|
|
|
|
|
# |
|
1330
|
|
|
|
|
|
|
|
|
1331
|
|
|
|
|
|
|
sub die_or_warn { |
|
1332
|
13
|
|
|
13
|
0
|
22
|
my $self = shift; |
|
1333
|
13
|
|
|
|
|
20
|
my $msg = shift; |
|
1334
|
|
|
|
|
|
|
|
|
1335
|
13
|
100
|
|
|
|
976
|
croak $msg if($self->{opt}->{strictmode}); |
|
1336
|
8
|
100
|
|
|
|
978
|
if(warnings::enabled()) { |
|
1337
|
5
|
|
|
|
|
948
|
carp "Warning: $msg"; |
|
1338
|
|
|
|
|
|
|
} |
|
1339
|
|
|
|
|
|
|
} |
|
1340
|
|
|
|
|
|
|
|
|
1341
|
|
|
|
|
|
|
|
|
1342
|
|
|
|
|
|
|
############################################################################## |
|
1343
|
|
|
|
|
|
|
# Method: new_hashref() |
|
1344
|
|
|
|
|
|
|
# |
|
1345
|
|
|
|
|
|
|
# This is a hook routine for overriding in a sub-class. Some people believe |
|
1346
|
|
|
|
|
|
|
# that using Tie::IxHash here will solve order-loss problems. |
|
1347
|
|
|
|
|
|
|
# |
|
1348
|
|
|
|
|
|
|
|
|
1349
|
|
|
|
|
|
|
sub new_hashref { |
|
1350
|
2385
|
|
|
2385
|
1
|
3263
|
my $self = shift; |
|
1351
|
|
|
|
|
|
|
|
|
1352
|
2385
|
|
|
|
|
9416
|
return { @_ }; |
|
1353
|
|
|
|
|
|
|
} |
|
1354
|
|
|
|
|
|
|
|
|
1355
|
|
|
|
|
|
|
|
|
1356
|
|
|
|
|
|
|
############################################################################## |
|
1357
|
|
|
|
|
|
|
# Method: collapse_content() |
|
1358
|
|
|
|
|
|
|
# |
|
1359
|
|
|
|
|
|
|
# Helper routine for array_to_hash |
|
1360
|
|
|
|
|
|
|
# |
|
1361
|
|
|
|
|
|
|
# Arguments expected are: |
|
1362
|
|
|
|
|
|
|
# - an XML::Simple object |
|
1363
|
|
|
|
|
|
|
# - a hashref |
|
1364
|
|
|
|
|
|
|
# the hashref is a former array, turned into a hash by array_to_hash because |
|
1365
|
|
|
|
|
|
|
# of the presence of key attributes |
|
1366
|
|
|
|
|
|
|
# at this point collapse_content avoids over-complicated structures like |
|
1367
|
|
|
|
|
|
|
# dir => { libexecdir => { content => '$exec_prefix/libexec' }, |
|
1368
|
|
|
|
|
|
|
# localstatedir => { content => '$prefix' }, |
|
1369
|
|
|
|
|
|
|
# } |
|
1370
|
|
|
|
|
|
|
# into |
|
1371
|
|
|
|
|
|
|
# dir => { libexecdir => '$exec_prefix/libexec', |
|
1372
|
|
|
|
|
|
|
# localstatedir => '$prefix', |
|
1373
|
|
|
|
|
|
|
# } |
|
1374
|
|
|
|
|
|
|
|
|
1375
|
|
|
|
|
|
|
sub collapse_content { |
|
1376
|
36
|
|
|
36
|
0
|
58
|
my $self = shift; |
|
1377
|
36
|
|
|
|
|
78
|
my $hashref = shift; |
|
1378
|
|
|
|
|
|
|
|
|
1379
|
36
|
|
|
|
|
78
|
my $contentkey = $self->{opt}->{contentkey}; |
|
1380
|
|
|
|
|
|
|
|
|
1381
|
|
|
|
|
|
|
# first go through the values,checking that they are fit to collapse |
|
1382
|
36
|
|
|
|
|
103
|
foreach my $val (values %$hashref) { |
|
1383
|
|
|
|
|
|
|
return $hashref unless ( (ref($val) eq 'HASH') |
|
1384
|
|
|
|
|
|
|
and (keys %$val == 1) |
|
1385
|
70
|
100
|
66
|
|
|
499
|
and (exists $val->{$contentkey}) |
|
|
|
|
66
|
|
|
|
|
|
1386
|
|
|
|
|
|
|
); |
|
1387
|
|
|
|
|
|
|
} |
|
1388
|
|
|
|
|
|
|
|
|
1389
|
|
|
|
|
|
|
# now collapse them |
|
1390
|
14
|
|
|
|
|
48
|
foreach my $key (keys %$hashref) { |
|
1391
|
48
|
|
|
|
|
182
|
$hashref->{$key}= $hashref->{$key}->{$contentkey}; |
|
1392
|
|
|
|
|
|
|
} |
|
1393
|
|
|
|
|
|
|
|
|
1394
|
14
|
|
|
|
|
39
|
return $hashref; |
|
1395
|
|
|
|
|
|
|
} |
|
1396
|
|
|
|
|
|
|
|
|
1397
|
|
|
|
|
|
|
|
|
1398
|
|
|
|
|
|
|
############################################################################## |
|
1399
|
|
|
|
|
|
|
# Method: value_to_xml() |
|
1400
|
|
|
|
|
|
|
# |
|
1401
|
|
|
|
|
|
|
# Helper routine for XMLout() - recurses through a data structure building up |
|
1402
|
|
|
|
|
|
|
# and returning an XML representation of that structure as a string. |
|
1403
|
|
|
|
|
|
|
# |
|
1404
|
|
|
|
|
|
|
# Arguments expected are: |
|
1405
|
|
|
|
|
|
|
# - the data structure to be encoded (usually a reference) |
|
1406
|
|
|
|
|
|
|
# - the XML tag name to use for this item |
|
1407
|
|
|
|
|
|
|
# - a string of spaces for use as the current indent level |
|
1408
|
|
|
|
|
|
|
# |
|
1409
|
|
|
|
|
|
|
|
|
1410
|
|
|
|
|
|
|
sub value_to_xml { |
|
1411
|
2186
|
|
|
2186
|
0
|
3062
|
my $self = shift;; |
|
1412
|
|
|
|
|
|
|
|
|
1413
|
|
|
|
|
|
|
|
|
1414
|
|
|
|
|
|
|
# Grab the other arguments |
|
1415
|
|
|
|
|
|
|
|
|
1416
|
2186
|
|
|
|
|
3553
|
my($ref, $name, $indent) = @_; |
|
1417
|
|
|
|
|
|
|
|
|
1418
|
2186
|
|
33
|
|
|
6932
|
my $named = (defined($name) and $name ne '' ? 1 : 0); |
|
1419
|
|
|
|
|
|
|
|
|
1420
|
2186
|
|
|
|
|
3012
|
my $nl = "\n"; |
|
1421
|
|
|
|
|
|
|
|
|
1422
|
2186
|
100
|
|
|
|
4204
|
my $is_root = $indent eq '' ? 1 : 0; # Warning, dirty hack! |
|
1423
|
2186
|
100
|
|
|
|
4788
|
if($self->{opt}->{noindent}) { |
|
1424
|
6
|
|
|
|
|
10
|
$indent = ''; |
|
1425
|
6
|
|
|
|
|
9
|
$nl = ''; |
|
1426
|
|
|
|
|
|
|
} |
|
1427
|
|
|
|
|
|
|
|
|
1428
|
|
|
|
|
|
|
|
|
1429
|
|
|
|
|
|
|
# Convert to XML |
|
1430
|
|
|
|
|
|
|
|
|
1431
|
2186
|
100
|
|
|
|
5928
|
if(my $refaddr = Scalar::Util::refaddr($ref)) { |
|
1432
|
|
|
|
|
|
|
croak "circular data structures not supported" |
|
1433
|
2170
|
100
|
|
|
|
5404
|
if $self->{_ancestors}->{$refaddr}; |
|
1434
|
2169
|
|
|
|
|
5217
|
$self->{_ancestors}->{$refaddr} = 1; |
|
1435
|
|
|
|
|
|
|
} |
|
1436
|
|
|
|
|
|
|
else { |
|
1437
|
16
|
100
|
|
|
|
29
|
if($named) { |
|
1438
|
|
|
|
|
|
|
return(join('', |
|
1439
|
|
|
|
|
|
|
$indent, '<', $name, '>', |
|
1440
|
14
|
100
|
|
|
|
51
|
($self->{opt}->{noescape} ? $ref : $self->escape_value($ref)), |
|
1441
|
|
|
|
|
|
|
'', $name, ">", $nl |
|
1442
|
|
|
|
|
|
|
)); |
|
1443
|
|
|
|
|
|
|
} |
|
1444
|
|
|
|
|
|
|
else { |
|
1445
|
2
|
|
|
|
|
7
|
return("$ref$nl"); |
|
1446
|
|
|
|
|
|
|
} |
|
1447
|
|
|
|
|
|
|
} |
|
1448
|
|
|
|
|
|
|
|
|
1449
|
|
|
|
|
|
|
|
|
1450
|
|
|
|
|
|
|
# Unfold hash to array if possible |
|
1451
|
|
|
|
|
|
|
|
|
1452
|
2169
|
100
|
100
|
|
|
18472
|
if(UNIVERSAL::isa($ref, 'HASH') # It is a hash |
|
|
|
|
66
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
1453
|
|
|
|
|
|
|
and keys %$ref # and it's not empty |
|
1454
|
|
|
|
|
|
|
and $self->{opt}->{keyattr} # and folding is enabled |
|
1455
|
|
|
|
|
|
|
and !$is_root # and its not the root element |
|
1456
|
|
|
|
|
|
|
) { |
|
1457
|
2003
|
|
|
|
|
4085
|
$ref = $self->hash_to_array($name, $ref); |
|
1458
|
|
|
|
|
|
|
} |
|
1459
|
|
|
|
|
|
|
|
|
1460
|
|
|
|
|
|
|
|
|
1461
|
2169
|
|
|
|
|
3554
|
my @result = (); |
|
1462
|
2169
|
|
|
|
|
2538
|
my($key, $value); |
|
1463
|
|
|
|
|
|
|
|
|
1464
|
|
|
|
|
|
|
|
|
1465
|
|
|
|
|
|
|
# Handle hashrefs |
|
1466
|
|
|
|
|
|
|
|
|
1467
|
2169
|
100
|
|
|
|
5905
|
if(UNIVERSAL::isa($ref, 'HASH')) { |
|
|
|
100
|
|
|
|
|
|
|
1468
|
|
|
|
|
|
|
|
|
1469
|
|
|
|
|
|
|
# Reintermediate grouped values if applicable |
|
1470
|
|
|
|
|
|
|
|
|
1471
|
1980
|
100
|
|
|
|
4541
|
if($self->{opt}->{grouptags}) { |
|
1472
|
12
|
|
|
|
|
33
|
$ref = $self->copy_hash($ref); |
|
1473
|
12
|
|
|
|
|
45
|
while(my($key, $val) = each %$ref) { |
|
1474
|
18
|
100
|
|
|
|
68
|
if($self->{opt}->{grouptags}->{$key}) { |
|
1475
|
|
|
|
|
|
|
$ref->{$key} = $self->new_hashref( |
|
1476
|
5
|
|
|
|
|
15
|
$self->{opt}->{grouptags}->{$key} => $val |
|
1477
|
|
|
|
|
|
|
); |
|
1478
|
|
|
|
|
|
|
} |
|
1479
|
|
|
|
|
|
|
} |
|
1480
|
|
|
|
|
|
|
} |
|
1481
|
|
|
|
|
|
|
|
|
1482
|
|
|
|
|
|
|
|
|
1483
|
|
|
|
|
|
|
# Scan for namespace declaration attributes |
|
1484
|
|
|
|
|
|
|
|
|
1485
|
1980
|
|
|
|
|
2679
|
my $nsdecls = ''; |
|
1486
|
1980
|
|
|
|
|
2242
|
my $default_ns_uri; |
|
1487
|
1980
|
100
|
|
|
|
3893
|
if($self->{nsup}) { |
|
1488
|
5
|
|
|
|
|
12
|
$ref = $self->copy_hash($ref); |
|
1489
|
5
|
|
|
|
|
21
|
$self->{nsup}->push_context(); |
|
1490
|
|
|
|
|
|
|
|
|
1491
|
|
|
|
|
|
|
# Look for default namespace declaration first |
|
1492
|
|
|
|
|
|
|
|
|
1493
|
5
|
100
|
|
|
|
47
|
if(exists($ref->{xmlns})) { |
|
1494
|
2
|
|
|
|
|
7
|
$self->{nsup}->declare_prefix('', $ref->{xmlns}); |
|
1495
|
2
|
|
|
|
|
36
|
$nsdecls .= qq( xmlns="$ref->{xmlns}"); |
|
1496
|
2
|
|
|
|
|
4
|
delete($ref->{xmlns}); |
|
1497
|
|
|
|
|
|
|
} |
|
1498
|
5
|
|
|
|
|
18
|
$default_ns_uri = $self->{nsup}->get_uri(''); |
|
1499
|
|
|
|
|
|
|
|
|
1500
|
|
|
|
|
|
|
|
|
1501
|
|
|
|
|
|
|
# Then check all the other keys |
|
1502
|
|
|
|
|
|
|
|
|
1503
|
5
|
|
|
|
|
40
|
foreach my $qname (keys(%$ref)) { |
|
1504
|
9
|
|
|
|
|
30
|
my($uri, $lname) = $self->{nsup}->parse_jclark_notation($qname); |
|
1505
|
9
|
100
|
|
|
|
100
|
if($uri) { |
|
1506
|
8
|
100
|
|
|
|
24
|
if($uri eq $xmlns_ns) { |
|
1507
|
1
|
|
|
|
|
5
|
$self->{nsup}->declare_prefix($lname, $ref->{$qname}); |
|
1508
|
1
|
|
|
|
|
23
|
$nsdecls .= qq( xmlns:$lname="$ref->{$qname}"); |
|
1509
|
1
|
|
|
|
|
3
|
delete($ref->{$qname}); |
|
1510
|
|
|
|
|
|
|
} |
|
1511
|
|
|
|
|
|
|
} |
|
1512
|
|
|
|
|
|
|
} |
|
1513
|
|
|
|
|
|
|
|
|
1514
|
|
|
|
|
|
|
# Translate any remaining Clarkian names |
|
1515
|
|
|
|
|
|
|
|
|
1516
|
5
|
|
|
|
|
16
|
foreach my $qname (keys(%$ref)) { |
|
1517
|
8
|
|
|
|
|
24
|
my($uri, $lname) = $self->{nsup}->parse_jclark_notation($qname); |
|
1518
|
8
|
100
|
|
|
|
73
|
if($uri) { |
|
1519
|
7
|
100
|
100
|
|
|
28
|
if($default_ns_uri and $uri eq $default_ns_uri) { |
|
1520
|
4
|
|
|
|
|
11
|
$ref->{$lname} = $ref->{$qname}; |
|
1521
|
4
|
|
|
|
|
12
|
delete($ref->{$qname}); |
|
1522
|
|
|
|
|
|
|
} |
|
1523
|
|
|
|
|
|
|
else { |
|
1524
|
3
|
|
|
|
|
12
|
my $prefix = $self->{nsup}->get_prefix($uri); |
|
1525
|
3
|
100
|
|
|
|
48
|
unless($prefix) { |
|
1526
|
|
|
|
|
|
|
# $self->{nsup}->declare_prefix(undef, $uri); |
|
1527
|
|
|
|
|
|
|
# $prefix = $self->{nsup}->get_prefix($uri); |
|
1528
|
1
|
|
|
|
|
3
|
$prefix = $self->{ns_prefix}++; |
|
1529
|
1
|
|
|
|
|
8
|
$self->{nsup}->declare_prefix($prefix, $uri); |
|
1530
|
1
|
|
|
|
|
21
|
$nsdecls .= qq( xmlns:$prefix="$uri"); |
|
1531
|
|
|
|
|
|
|
} |
|
1532
|
3
|
|
|
|
|
10
|
$ref->{"$prefix:$lname"} = $ref->{$qname}; |
|
1533
|
3
|
|
|
|
|
7
|
delete($ref->{$qname}); |
|
1534
|
|
|
|
|
|
|
} |
|
1535
|
|
|
|
|
|
|
} |
|
1536
|
|
|
|
|
|
|
} |
|
1537
|
|
|
|
|
|
|
} |
|
1538
|
|
|
|
|
|
|
|
|
1539
|
|
|
|
|
|
|
|
|
1540
|
1980
|
|
|
|
|
2989
|
my @nested = (); |
|
1541
|
1980
|
|
|
|
|
2462
|
my $text_content = undef; |
|
1542
|
1980
|
100
|
|
|
|
3968
|
if($named) { |
|
1543
|
1977
|
|
|
|
|
4342
|
push @result, $indent, '<', $name, $nsdecls; |
|
1544
|
|
|
|
|
|
|
} |
|
1545
|
|
|
|
|
|
|
|
|
1546
|
1980
|
100
|
|
|
|
3732
|
if(keys %$ref) { |
|
1547
|
1977
|
|
|
|
|
2548
|
my $first_arg = 1; |
|
1548
|
1977
|
|
|
|
|
4457
|
foreach my $key ($self->sorted_keys($name, $ref)) { |
|
1549
|
3948
|
|
|
|
|
6630
|
my $value = $ref->{$key}; |
|
1550
|
3948
|
100
|
|
|
|
9041
|
next if(substr($key, 0, 1) eq '-'); |
|
1551
|
3946
|
100
|
|
|
|
8487
|
if(!defined($value)) { |
|
1552
|
5
|
100
|
|
|
|
14
|
next if $self->{opt}->{suppressempty}; |
|
1553
|
4
|
100
|
66
|
|
|
20
|
unless(exists($self->{opt}->{suppressempty}) |
|
1554
|
|
|
|
|
|
|
and !defined($self->{opt}->{suppressempty}) |
|
1555
|
|
|
|
|
|
|
) { |
|
1556
|
2
|
100
|
|
|
|
315
|
carp 'Use of uninitialized value' if warnings::enabled(); |
|
1557
|
|
|
|
|
|
|
} |
|
1558
|
4
|
100
|
|
|
|
40
|
if($key eq $self->{opt}->{contentkey}) { |
|
1559
|
1
|
|
|
|
|
3
|
$text_content = ''; |
|
1560
|
|
|
|
|
|
|
} |
|
1561
|
|
|
|
|
|
|
else { |
|
1562
|
3
|
100
|
|
|
|
11
|
$value = exists($self->{opt}->{suppressempty}) ? {} : ''; |
|
1563
|
|
|
|
|
|
|
} |
|
1564
|
|
|
|
|
|
|
} |
|
1565
|
|
|
|
|
|
|
|
|
1566
|
3945
|
100
|
66
|
|
|
13611
|
if(!ref($value) |
|
|
|
|
66
|
|
|
|
|
|
1567
|
|
|
|
|
|
|
and $self->{opt}->{valueattr} |
|
1568
|
|
|
|
|
|
|
and $self->{opt}->{valueattr}->{$key} |
|
1569
|
|
|
|
|
|
|
) { |
|
1570
|
|
|
|
|
|
|
$value = $self->new_hashref( |
|
1571
|
2
|
|
|
|
|
7
|
$self->{opt}->{valueattr}->{$key} => $value |
|
1572
|
|
|
|
|
|
|
); |
|
1573
|
|
|
|
|
|
|
} |
|
1574
|
|
|
|
|
|
|
|
|
1575
|
3945
|
100
|
66
|
|
|
12031
|
if(ref($value) or $self->{opt}->{noattr}) { |
|
1576
|
187
|
|
|
|
|
701
|
push @nested, |
|
1577
|
|
|
|
|
|
|
$self->value_to_xml($value, $key, "$indent "); |
|
1578
|
|
|
|
|
|
|
} |
|
1579
|
|
|
|
|
|
|
else { |
|
1580
|
3758
|
100
|
|
|
|
8208
|
if($key eq $self->{opt}->{contentkey}) { |
|
1581
|
19
|
50
|
|
|
|
65
|
$value = $self->escape_value($value) unless($self->{opt}->{noescape}); |
|
1582
|
19
|
|
|
|
|
50
|
$text_content = $value; |
|
1583
|
|
|
|
|
|
|
} |
|
1584
|
|
|
|
|
|
|
else { |
|
1585
|
3739
|
100
|
|
|
|
11506
|
$value = $self->escape_attr($value) unless($self->{opt}->{noescape}); |
|
1586
|
|
|
|
|
|
|
push @result, "\n$indent " . ' ' x length($name) |
|
1587
|
3739
|
100
|
100
|
|
|
10534
|
if($self->{opt}->{attrindent} and !$first_arg); |
|
1588
|
3739
|
|
|
|
|
8362
|
push @result, ' ', $key, '="', $value , '"'; |
|
1589
|
3739
|
|
|
|
|
7291
|
$first_arg = 0; |
|
1590
|
|
|
|
|
|
|
} |
|
1591
|
|
|
|
|
|
|
} |
|
1592
|
|
|
|
|
|
|
} |
|
1593
|
|
|
|
|
|
|
} |
|
1594
|
|
|
|
|
|
|
else { |
|
1595
|
3
|
|
|
|
|
6
|
$text_content = ''; |
|
1596
|
|
|
|
|
|
|
} |
|
1597
|
|
|
|
|
|
|
|
|
1598
|
1979
|
100
|
100
|
|
|
8430
|
if(@nested or defined($text_content)) { |
|
1599
|
180
|
100
|
|
|
|
329
|
if($named) { |
|
1600
|
177
|
|
|
|
|
284
|
push @result, ">"; |
|
1601
|
177
|
100
|
|
|
|
316
|
if(defined($text_content)) { |
|
1602
|
22
|
|
|
|
|
37
|
push @result, $text_content; |
|
1603
|
22
|
50
|
|
|
|
72
|
$nested[0] =~ s/^\s+// if(@nested); |
|
1604
|
|
|
|
|
|
|
} |
|
1605
|
|
|
|
|
|
|
else { |
|
1606
|
155
|
|
|
|
|
240
|
push @result, $nl; |
|
1607
|
|
|
|
|
|
|
} |
|
1608
|
177
|
100
|
|
|
|
394
|
if(@nested) { |
|
1609
|
155
|
|
|
|
|
296
|
push @result, @nested, $indent; |
|
1610
|
|
|
|
|
|
|
} |
|
1611
|
177
|
|
|
|
|
372
|
push @result, '', $name, ">", $nl; |
|
1612
|
|
|
|
|
|
|
} |
|
1613
|
|
|
|
|
|
|
else { |
|
1614
|
3
|
|
|
|
|
6
|
push @result, @nested; # Special case if no root elements |
|
1615
|
|
|
|
|
|
|
} |
|
1616
|
|
|
|
|
|
|
} |
|
1617
|
|
|
|
|
|
|
else { |
|
1618
|
1799
|
|
|
|
|
3125
|
push @result, " />", $nl; |
|
1619
|
|
|
|
|
|
|
} |
|
1620
|
1979
|
100
|
|
|
|
5605
|
$self->{nsup}->pop_context() if($self->{nsup}); |
|
1621
|
|
|
|
|
|
|
} |
|
1622
|
|
|
|
|
|
|
|
|
1623
|
|
|
|
|
|
|
|
|
1624
|
|
|
|
|
|
|
# Handle arrayrefs |
|
1625
|
|
|
|
|
|
|
|
|
1626
|
|
|
|
|
|
|
elsif(UNIVERSAL::isa($ref, 'ARRAY')) { |
|
1627
|
188
|
|
|
|
|
364
|
foreach $value (@$ref) { |
|
1628
|
2851
|
100
|
66
|
|
|
5801
|
next if !defined($value) and $self->{opt}->{suppressempty}; |
|
1629
|
2850
|
100
|
|
|
|
9313
|
if(!ref($value)) { |
|
|
|
100
|
|
|
|
|
|
|
1630
|
|
|
|
|
|
|
push @result, |
|
1631
|
|
|
|
|
|
|
$indent, '<', $name, '>', |
|
1632
|
930
|
100
|
|
|
|
2840
|
($self->{opt}->{noescape} ? $value : $self->escape_value($value)), |
|
1633
|
|
|
|
|
|
|
'', $name, ">$nl"; |
|
1634
|
|
|
|
|
|
|
} |
|
1635
|
|
|
|
|
|
|
elsif(UNIVERSAL::isa($value, 'HASH')) { |
|
1636
|
1877
|
|
|
|
|
4168
|
push @result, $self->value_to_xml($value, $name, $indent); |
|
1637
|
|
|
|
|
|
|
} |
|
1638
|
|
|
|
|
|
|
else { |
|
1639
|
43
|
|
|
|
|
136
|
push @result, |
|
1640
|
|
|
|
|
|
|
$indent, '<', $name, ">$nl", |
|
1641
|
|
|
|
|
|
|
$self->value_to_xml($value, 'anon', "$indent "), |
|
1642
|
|
|
|
|
|
|
$indent, '', $name, ">$nl"; |
|
1643
|
|
|
|
|
|
|
} |
|
1644
|
|
|
|
|
|
|
} |
|
1645
|
|
|
|
|
|
|
} |
|
1646
|
|
|
|
|
|
|
|
|
1647
|
|
|
|
|
|
|
else { |
|
1648
|
1
|
|
|
|
|
118
|
croak "Can't encode a value of type: " . ref($ref); |
|
1649
|
|
|
|
|
|
|
} |
|
1650
|
|
|
|
|
|
|
|
|
1651
|
|
|
|
|
|
|
|
|
1652
|
2167
|
50
|
|
|
|
6140
|
if(my $refaddr = Scalar::Util::refaddr($ref)) { |
|
1653
|
2167
|
|
|
|
|
4414
|
delete $self->{_ancestors}->{$refaddr}; |
|
1654
|
|
|
|
|
|
|
} |
|
1655
|
|
|
|
|
|
|
|
|
1656
|
2167
|
|
|
|
|
13059
|
return(join('', @result)); |
|
1657
|
|
|
|
|
|
|
} |
|
1658
|
|
|
|
|
|
|
|
|
1659
|
|
|
|
|
|
|
|
|
1660
|
|
|
|
|
|
|
############################################################################## |
|
1661
|
|
|
|
|
|
|
# Method: sorted_keys() |
|
1662
|
|
|
|
|
|
|
# |
|
1663
|
|
|
|
|
|
|
# Returns the keys of the referenced hash sorted into alphabetical order, but |
|
1664
|
|
|
|
|
|
|
# with the 'key' key (as in KeyAttr) first, if there is one. |
|
1665
|
|
|
|
|
|
|
# |
|
1666
|
|
|
|
|
|
|
|
|
1667
|
|
|
|
|
|
|
sub sorted_keys { |
|
1668
|
1977
|
|
|
1977
|
1
|
3097
|
my($self, $name, $ref) = @_; |
|
1669
|
|
|
|
|
|
|
|
|
1670
|
1977
|
50
|
|
|
|
4295
|
return keys %$ref if $self->{opt}->{nosort}; |
|
1671
|
|
|
|
|
|
|
|
|
1672
|
1977
|
|
|
|
|
5560
|
my %hash = %$ref; |
|
1673
|
1977
|
|
|
|
|
3555
|
my $keyattr = $self->{opt}->{keyattr}; |
|
1674
|
|
|
|
|
|
|
|
|
1675
|
1977
|
|
|
|
|
2496
|
my @key; |
|
1676
|
|
|
|
|
|
|
|
|
1677
|
1977
|
100
|
|
|
|
4928
|
if(ref $keyattr eq 'HASH') { |
|
|
|
100
|
|
|
|
|
|
|
1678
|
1877
|
100
|
100
|
|
|
9282
|
if(exists $keyattr->{$name} and exists $hash{$keyattr->{$name}->[0]}) { |
|
1679
|
1847
|
|
|
|
|
3383
|
push @key, $keyattr->{$name}->[0]; |
|
1680
|
1847
|
|
|
|
|
3426
|
delete $hash{$keyattr->{$name}->[0]}; |
|
1681
|
|
|
|
|
|
|
} |
|
1682
|
|
|
|
|
|
|
} |
|
1683
|
|
|
|
|
|
|
elsif(ref $keyattr eq 'ARRAY') { |
|
1684
|
95
|
|
|
|
|
119
|
foreach (@{$keyattr}) { |
|
|
95
|
|
|
|
|
214
|
|
|
1685
|
237
|
100
|
|
|
|
585
|
if(exists $hash{$_}) { |
|
1686
|
21
|
|
|
|
|
37
|
push @key, $_; |
|
1687
|
21
|
|
|
|
|
34
|
delete $hash{$_}; |
|
1688
|
21
|
|
|
|
|
32
|
last; |
|
1689
|
|
|
|
|
|
|
} |
|
1690
|
|
|
|
|
|
|
} |
|
1691
|
|
|
|
|
|
|
} |
|
1692
|
|
|
|
|
|
|
|
|
1693
|
1977
|
|
|
|
|
7403
|
return(@key, sort keys %hash); |
|
1694
|
|
|
|
|
|
|
} |
|
1695
|
|
|
|
|
|
|
|
|
1696
|
|
|
|
|
|
|
############################################################################## |
|
1697
|
|
|
|
|
|
|
# Method: escape_value() |
|
1698
|
|
|
|
|
|
|
# |
|
1699
|
|
|
|
|
|
|
# Helper routine for automatically escaping values for XMLout(). |
|
1700
|
|
|
|
|
|
|
# Expects a scalar data value. Returns escaped version. |
|
1701
|
|
|
|
|
|
|
# |
|
1702
|
|
|
|
|
|
|
|
|
1703
|
|
|
|
|
|
|
sub escape_value { |
|
1704
|
4697
|
|
|
4697
|
1
|
6847
|
my($self, $data) = @_; |
|
1705
|
|
|
|
|
|
|
|
|
1706
|
4697
|
100
|
|
|
|
9017
|
return '' unless(defined($data)); |
|
1707
|
|
|
|
|
|
|
|
|
1708
|
4695
|
|
|
|
|
7496
|
$data =~ s/&/&/sg; |
|
1709
|
4695
|
|
|
|
|
6491
|
$data =~ s/</sg; |
|
1710
|
4695
|
|
|
|
|
6702
|
$data =~ s/>/>/sg; |
|
1711
|
4695
|
|
|
|
|
6620
|
$data =~ s/"/"/sg; |
|
1712
|
|
|
|
|
|
|
|
|
1713
|
4695
|
100
|
|
|
|
17672
|
my $level = $self->{opt}->{numericescape} or return $data; |
|
1714
|
|
|
|
|
|
|
|
|
1715
|
4
|
|
|
|
|
11
|
return $self->numeric_escape($data, $level); |
|
1716
|
|
|
|
|
|
|
} |
|
1717
|
|
|
|
|
|
|
|
|
1718
|
|
|
|
|
|
|
sub numeric_escape { |
|
1719
|
4
|
|
|
4
|
1
|
9
|
my($self, $data, $level) = @_; |
|
1720
|
|
|
|
|
|
|
|
|
1721
|
4
|
100
|
|
|
|
11
|
if($self->{opt}->{numericescape} eq '2') { |
|
1722
|
2
|
|
|
|
|
10
|
$data =~ s/([^\x00-\x7F])/'' . ord($1) . ';'/gse; |
|
|
2
|
|
|
|
|
7
|
|
|
1723
|
|
|
|
|
|
|
} |
|
1724
|
|
|
|
|
|
|
else { |
|
1725
|
2
|
|
|
|
|
8
|
$data =~ s/([^\x00-\xFF])/'' . ord($1) . ';'/gse; |
|
|
1
|
|
|
|
|
4
|
|
|
1726
|
|
|
|
|
|
|
} |
|
1727
|
|
|
|
|
|
|
|
|
1728
|
4
|
|
|
|
|
14
|
return $data; |
|
1729
|
|
|
|
|
|
|
} |
|
1730
|
|
|
|
|
|
|
|
|
1731
|
|
|
|
|
|
|
############################################################################## |
|
1732
|
|
|
|
|
|
|
# Method: escape_attr() |
|
1733
|
|
|
|
|
|
|
# |
|
1734
|
|
|
|
|
|
|
# Helper routine for escaping attribute values. Defaults to escape_value(), |
|
1735
|
|
|
|
|
|
|
# but may be overridden by a subclass to customise behaviour. |
|
1736
|
|
|
|
|
|
|
# |
|
1737
|
|
|
|
|
|
|
|
|
1738
|
|
|
|
|
|
|
sub escape_attr { |
|
1739
|
3737
|
|
|
3737
|
1
|
4580
|
my $self = shift; |
|
1740
|
|
|
|
|
|
|
|
|
1741
|
3737
|
|
|
|
|
8266
|
return $self->escape_value(@_); |
|
1742
|
|
|
|
|
|
|
} |
|
1743
|
|
|
|
|
|
|
|
|
1744
|
|
|
|
|
|
|
|
|
1745
|
|
|
|
|
|
|
############################################################################## |
|
1746
|
|
|
|
|
|
|
# Method: hash_to_array() |
|
1747
|
|
|
|
|
|
|
# |
|
1748
|
|
|
|
|
|
|
# Helper routine for value_to_xml(). |
|
1749
|
|
|
|
|
|
|
# Attempts to 'unfold' a hash of hashes into an array of hashes. Returns a |
|
1750
|
|
|
|
|
|
|
# reference to the array on success or the original hash if unfolding is |
|
1751
|
|
|
|
|
|
|
# not possible. |
|
1752
|
|
|
|
|
|
|
# |
|
1753
|
|
|
|
|
|
|
|
|
1754
|
|
|
|
|
|
|
sub hash_to_array { |
|
1755
|
2003
|
|
|
2003
|
0
|
2713
|
my $self = shift; |
|
1756
|
2003
|
|
|
|
|
3424
|
my $parent = shift; |
|
1757
|
2003
|
|
|
|
|
2901
|
my $hashref = shift; |
|
1758
|
|
|
|
|
|
|
|
|
1759
|
2003
|
|
|
|
|
3120
|
my $arrayref = []; |
|
1760
|
|
|
|
|
|
|
|
|
1761
|
2003
|
|
|
|
|
2761
|
my($key, $value); |
|
1762
|
|
|
|
|
|
|
|
|
1763
|
2003
|
50
|
|
|
|
8427
|
my @keys = $self->{opt}->{nosort} ? keys %$hashref : sort keys %$hashref; |
|
1764
|
2003
|
|
|
|
|
3601
|
foreach $key (@keys) { |
|
1765
|
3844
|
|
|
|
|
6327
|
$value = $hashref->{$key}; |
|
1766
|
3844
|
100
|
|
|
|
15857
|
return($hashref) unless(UNIVERSAL::isa($value, 'HASH')); |
|
1767
|
|
|
|
|
|
|
|
|
1768
|
1948
|
100
|
|
|
|
4719
|
if(ref($self->{opt}->{keyattr}) eq 'HASH') { |
|
1769
|
1931
|
100
|
|
|
|
4092
|
return($hashref) unless(defined($self->{opt}->{keyattr}->{$parent})); |
|
1770
|
|
|
|
|
|
|
push @$arrayref, $self->copy_hash( |
|
1771
|
1929
|
|
|
|
|
4849
|
$value, $self->{opt}->{keyattr}->{$parent}->[0] => $key |
|
1772
|
|
|
|
|
|
|
); |
|
1773
|
|
|
|
|
|
|
} |
|
1774
|
|
|
|
|
|
|
else { |
|
1775
|
17
|
|
|
|
|
84
|
push(@$arrayref, { $self->{opt}->{keyattr}->[0] => $key, %$value }); |
|
1776
|
|
|
|
|
|
|
} |
|
1777
|
|
|
|
|
|
|
} |
|
1778
|
|
|
|
|
|
|
|
|
1779
|
105
|
|
|
|
|
344
|
return($arrayref); |
|
1780
|
|
|
|
|
|
|
} |
|
1781
|
|
|
|
|
|
|
|
|
1782
|
|
|
|
|
|
|
|
|
1783
|
|
|
|
|
|
|
############################################################################## |
|
1784
|
|
|
|
|
|
|
# Method: copy_hash() |
|
1785
|
|
|
|
|
|
|
# |
|
1786
|
|
|
|
|
|
|
# Helper routine for hash_to_array(). When unfolding a hash of hashes into |
|
1787
|
|
|
|
|
|
|
# an array of hashes, we need to copy the key from the outer hash into the |
|
1788
|
|
|
|
|
|
|
# inner hash. This routine makes a copy of the original hash so we don't |
|
1789
|
|
|
|
|
|
|
# destroy the original data structure. You might wish to override this |
|
1790
|
|
|
|
|
|
|
# method if you're using tied hashes and don't want them to get untied. |
|
1791
|
|
|
|
|
|
|
# |
|
1792
|
|
|
|
|
|
|
|
|
1793
|
|
|
|
|
|
|
sub copy_hash { |
|
1794
|
1946
|
|
|
1946
|
1
|
4189
|
my($self, $orig, @extra) = @_; |
|
1795
|
|
|
|
|
|
|
|
|
1796
|
1946
|
|
|
|
|
8976
|
return { @extra, %$orig }; |
|
1797
|
|
|
|
|
|
|
} |
|
1798
|
|
|
|
|
|
|
|
|
1799
|
|
|
|
|
|
|
############################################################################## |
|
1800
|
|
|
|
|
|
|
# Methods required for building trees from SAX events |
|
1801
|
|
|
|
|
|
|
############################################################################## |
|
1802
|
|
|
|
|
|
|
|
|
1803
|
|
|
|
|
|
|
sub start_document { |
|
1804
|
161
|
|
|
161
|
0
|
32230
|
my $self = shift; |
|
1805
|
|
|
|
|
|
|
|
|
1806
|
161
|
100
|
|
|
|
515
|
$self->handle_options('in') unless($self->{opt}); |
|
1807
|
|
|
|
|
|
|
|
|
1808
|
161
|
|
|
|
|
340
|
$self->{lists} = []; |
|
1809
|
161
|
|
|
|
|
642
|
$self->{curlist} = $self->{tree} = []; |
|
1810
|
|
|
|
|
|
|
} |
|
1811
|
|
|
|
|
|
|
|
|
1812
|
|
|
|
|
|
|
|
|
1813
|
|
|
|
|
|
|
sub start_element { |
|
1814
|
3531
|
|
|
3531
|
0
|
2176334
|
my $self = shift; |
|
1815
|
3531
|
|
|
|
|
4990
|
my $element = shift; |
|
1816
|
|
|
|
|
|
|
|
|
1817
|
3531
|
|
|
|
|
5718
|
my $name = $element->{Name}; |
|
1818
|
3531
|
100
|
|
|
|
9035
|
if($self->{opt}->{nsexpand}) { |
|
1819
|
11
|
|
50
|
|
|
30
|
$name = $element->{LocalName} || ''; |
|
1820
|
11
|
100
|
|
|
|
25
|
if($element->{NamespaceURI}) { |
|
1821
|
6
|
|
|
|
|
19
|
$name = '{' . $element->{NamespaceURI} . '}' . $name; |
|
1822
|
|
|
|
|
|
|
} |
|
1823
|
|
|
|
|
|
|
} |
|
1824
|
3531
|
|
|
|
|
5626
|
my $attributes = {}; |
|
1825
|
3531
|
50
|
|
|
|
9007
|
if($element->{Attributes}) { # Might be undef |
|
1826
|
3531
|
|
|
|
|
4022
|
foreach my $attr (values %{$element->{Attributes}}) { |
|
|
3531
|
|
|
|
|
9548
|
|
|
1827
|
4144
|
100
|
|
|
|
8828
|
if($self->{opt}->{nsexpand}) { |
|
1828
|
6
|
|
50
|
|
|
17
|
my $name = $attr->{LocalName} || ''; |
|
1829
|
6
|
100
|
|
|
|
13
|
if($attr->{NamespaceURI}) { |
|
1830
|
4
|
|
|
|
|
12
|
$name = '{' . $attr->{NamespaceURI} . '}' . $name |
|
1831
|
|
|
|
|
|
|
} |
|
1832
|
6
|
50
|
|
|
|
15
|
$name = 'xmlns' if($name eq $bad_def_ns_jcn); |
|
1833
|
6
|
|
|
|
|
20
|
$attributes->{$name} = $attr->{Value}; |
|
1834
|
|
|
|
|
|
|
} |
|
1835
|
|
|
|
|
|
|
else { |
|
1836
|
4138
|
|
|
|
|
13381
|
$attributes->{$attr->{Name}} = $attr->{Value}; |
|
1837
|
|
|
|
|
|
|
} |
|
1838
|
|
|
|
|
|
|
} |
|
1839
|
|
|
|
|
|
|
} |
|
1840
|
3531
|
|
|
|
|
6954
|
my $newlist = [ $attributes ]; |
|
1841
|
3531
|
|
|
|
|
4549
|
push @{ $self->{lists} }, $self->{curlist}; |
|
|
3531
|
|
|
|
|
7660
|
|
|
1842
|
3531
|
|
|
|
|
4279
|
push @{ $self->{curlist} }, $name => $newlist; |
|
|
3531
|
|
|
|
|
7719
|
|
|
1843
|
3531
|
|
|
|
|
11133
|
$self->{curlist} = $newlist; |
|
1844
|
|
|
|
|
|
|
} |
|
1845
|
|
|
|
|
|
|
|
|
1846
|
|
|
|
|
|
|
|
|
1847
|
|
|
|
|
|
|
sub characters { |
|
1848
|
4943
|
|
|
4943
|
0
|
285079
|
my $self = shift; |
|
1849
|
4943
|
|
|
|
|
6379
|
my $chars = shift; |
|
1850
|
|
|
|
|
|
|
|
|
1851
|
4943
|
|
|
|
|
7624
|
my $text = $chars->{Data}; |
|
1852
|
4943
|
|
|
|
|
7370
|
my $clist = $self->{curlist}; |
|
1853
|
4943
|
|
|
|
|
7263
|
my $pos = $#$clist; |
|
1854
|
|
|
|
|
|
|
|
|
1855
|
4943
|
100
|
100
|
|
|
21826
|
if ($pos > 0 and $clist->[$pos - 1] eq '0') { |
|
1856
|
11
|
|
|
|
|
37
|
$clist->[$pos] .= $text; |
|
1857
|
|
|
|
|
|
|
} |
|
1858
|
|
|
|
|
|
|
else { |
|
1859
|
4932
|
|
|
|
|
18486
|
push @$clist, 0 => $text; |
|
1860
|
|
|
|
|
|
|
} |
|
1861
|
|
|
|
|
|
|
} |
|
1862
|
|
|
|
|
|
|
|
|
1863
|
|
|
|
|
|
|
|
|
1864
|
|
|
|
|
|
|
sub end_element { |
|
1865
|
3531
|
|
|
3531
|
0
|
252681
|
my $self = shift; |
|
1866
|
|
|
|
|
|
|
|
|
1867
|
3531
|
|
|
|
|
4331
|
$self->{curlist} = pop @{ $self->{lists} }; |
|
|
3531
|
|
|
|
|
11940
|
|
|
1868
|
|
|
|
|
|
|
} |
|
1869
|
|
|
|
|
|
|
|
|
1870
|
|
|
|
|
|
|
|
|
1871
|
|
|
|
|
|
|
sub end_document { |
|
1872
|
161
|
|
|
161
|
0
|
16029
|
my $self = shift; |
|
1873
|
|
|
|
|
|
|
|
|
1874
|
161
|
|
|
|
|
316
|
delete($self->{curlist}); |
|
1875
|
161
|
|
|
|
|
333
|
delete($self->{lists}); |
|
1876
|
|
|
|
|
|
|
|
|
1877
|
161
|
|
|
|
|
234
|
my $tree = $self->{tree}; |
|
1878
|
161
|
|
|
|
|
311
|
delete($self->{tree}); |
|
1879
|
|
|
|
|
|
|
|
|
1880
|
|
|
|
|
|
|
|
|
1881
|
|
|
|
|
|
|
# Return tree as-is to XMLin() |
|
1882
|
|
|
|
|
|
|
|
|
1883
|
161
|
100
|
|
|
|
1998
|
return($tree) if($self->{nocollapse}); |
|
1884
|
|
|
|
|
|
|
|
|
1885
|
|
|
|
|
|
|
|
|
1886
|
|
|
|
|
|
|
# Or collapse it before returning it to SAX parser class |
|
1887
|
|
|
|
|
|
|
|
|
1888
|
7
|
100
|
|
|
|
18
|
if($self->{opt}->{keeproot}) { |
|
1889
|
1
|
|
|
|
|
5
|
$tree = $self->collapse({}, @$tree); |
|
1890
|
|
|
|
|
|
|
} |
|
1891
|
|
|
|
|
|
|
else { |
|
1892
|
6
|
|
|
|
|
8
|
$tree = $self->collapse(@{$tree->[1]}); |
|
|
6
|
|
|
|
|
24
|
|
|
1893
|
|
|
|
|
|
|
} |
|
1894
|
|
|
|
|
|
|
|
|
1895
|
7
|
100
|
|
|
|
57
|
if($self->{opt}->{datahandler}) { |
|
1896
|
2
|
|
|
|
|
7
|
return($self->{opt}->{datahandler}->($self, $tree)); |
|
1897
|
|
|
|
|
|
|
} |
|
1898
|
|
|
|
|
|
|
|
|
1899
|
5
|
|
|
|
|
163
|
return($tree); |
|
1900
|
|
|
|
|
|
|
} |
|
1901
|
|
|
|
|
|
|
|
|
1902
|
|
|
|
|
|
|
*xml_in = \&XMLin; |
|
1903
|
|
|
|
|
|
|
*xml_out = \&XMLout; |
|
1904
|
|
|
|
|
|
|
|
|
1905
|
|
|
|
|
|
|
1; |
|
1906
|
|
|
|
|
|
|
|
|
1907
|
|
|
|
|
|
|
__END__ |