blib/lib/XML/ExtOn.pm | |||
---|---|---|---|
Criterion | Covered | Total | % |
statement | 31 | 33 | 93.9 |
branch | n/a | ||
condition | n/a | ||
subroutine | 11 | 11 | 100.0 |
pod | n/a | ||
total | 42 | 44 | 95.4 |
line | stmt | bran | cond | sub | pod | time | code |
---|---|---|---|---|---|---|---|
1 | package XML::ExtOn; | ||||||
2 | |||||||
3 | #$Id: ExtOn.pm 966 2011-08-07 18:07:19Z zag $ | ||||||
4 | |||||||
5 | =pod | ||||||
6 | |||||||
7 | =head1 NAME | ||||||
8 | |||||||
9 | XML::ExtOn - The handler for expansion of Perl SAX by objects. | ||||||
10 | |||||||
11 | =head1 SYNOPSYS | ||||||
12 | |||||||
13 | use XML::ExtOn; | ||||||
14 | |||||||
15 | For write XML: | ||||||
16 | |||||||
17 | use XML::ExtOn; | ||||||
18 | my $buf; | ||||||
19 | my $wrt = XML::ExtOn::Writer->new( Output => \$buf ); | ||||||
20 | my $ex_parser = new XML::ExtOn:: Handler => $wrt; | ||||||
21 | $ex_parser->start_document; | ||||||
22 | my $root = $ex_parser->mk_element("Root"); | ||||||
23 | $root->add_namespace( | ||||||
24 | "myns" => 'http://example.com/myns', | ||||||
25 | "myns_test", 'http://example.com/myns_test' | ||||||
26 | ); | ||||||
27 | $ex_parser->start_element( $root ); | ||||||
28 | my $el = $root->mk_element('vars'); | ||||||
29 | %{ $el->attrs_by_prefix("myns") } = ( v1 => 1, v2 => 3 ); | ||||||
30 | %{ $el->attrs_by_prefix("myns_test") } = | ||||||
31 | ( var1 => "test ns", var2 => "2333" ); | ||||||
32 | $root->add_content($el); | ||||||
33 | $ex_parser->end_element; | ||||||
34 | $ex_parser->end_document; | ||||||
35 | print $buf; | ||||||
36 | |||||||
37 | Result: | ||||||
38 | |||||||
39 | |||||||
40 | | ||||||
41 | xmlns:myns_test="http://example.com/myns_test"> | ||||||
42 | | ||||||
43 | myns_test:var1="test ns" | ||||||
44 | myns:v1="1" myns:v2="3"/> | ||||||
45 | |||||||
46 | |||||||
47 | For handle events | ||||||
48 | |||||||
49 | use base 'XML::ExtOn'; | ||||||
50 | |||||||
51 | Begin method for handle SAX event start_element: | ||||||
52 | |||||||
53 | sub on_start_element { | ||||||
54 | my ( $self, $elem ) = @_; | ||||||
55 | |||||||
56 | ... | ||||||
57 | |||||||
58 | Check localname for element and add tag C |
||||||
59 | |||||||
60 | if ( $elem->local_name eq 'gallery' ) { | ||||||
61 | $elem->add_content( | ||||||
62 | $self->mk_element('image')->add_content( | ||||||
63 | $self->mk_characters( "Image number: $_" ) | ||||||
64 | ) | ||||||
65 | ) for 1..2 ; | ||||||
66 | } | ||||||
67 | |||||||
68 | XML Before: | ||||||
69 | |||||||
70 | |||||||
71 | |
||||||
72 | |
||||||
73 | |||||||
74 | |||||||
75 | After: | ||||||
76 | |||||||
77 | |||||||
78 | |
||||||
79 | |
||||||
80 | |
||||||
81 | |
||||||
82 | |||||||
83 | |||||||
84 | |||||||
85 | Register namespace and set variables | ||||||
86 | |||||||
87 | $elem->add_namespace('demons','http://example.org/demo_namespace'); | ||||||
88 | $elem->add_namespace('ns2','http://example.org/ns2'); | ||||||
89 | #set attributes for name space | ||||||
90 | my $demo_attrs = $elem->attrs_by_prefix('demons'); | ||||||
91 | %{$demo_attrs} = ( variable1=>1, 'variable2'=>2); | ||||||
92 | #set attributes for namespace URI | ||||||
93 | my $ns2_attrs = $elem->attrs_by_ns_uri('http://example.org/ns2'); | ||||||
94 | %{$ns2_attrs} = ( var=> 'ns1', 'raw'=>2); | ||||||
95 | |||||||
96 | Result: | ||||||
97 | |||||||
98 | |||||||
99 | xmlns:ns2="http://example.org/ns2" | ||||||
100 | demons:variable2="2" ns2:var="ns1" | ||||||
101 | demons:variable1="1" ns2:raw="2"/> | ||||||
102 | |||||||
103 | Delete content of element | ||||||
104 | |||||||
105 | if ( $elem->local_name eq 'demo_delete') { | ||||||
106 | $elem->skip_content | ||||||
107 | } | ||||||
108 | |||||||
109 | XML before: | ||||||
110 | |||||||
111 | |||||||
112 | |
||||||
113 | |
||||||
114 | text |
||||||
115 | |||||||
116 | |||||||
117 | |||||||
118 | After: | ||||||
119 | |||||||
120 | |||||||
121 | |
||||||
122 | |
||||||
123 | |||||||
124 | |||||||
125 | Add XML: | ||||||
126 | |||||||
127 | $elem->add_content ( | ||||||
128 | $self->mk_from_xml(' text |
||||||
129 | ) | ||||||
130 | Can add element after current | ||||||
131 | |||||||
132 | ... | ||||||
133 | return [ $elem, $self->mk_element("after") ]; | ||||||
134 | } | ||||||
135 | |||||||
136 | =head1 DESCRIPTION | ||||||
137 | |||||||
138 | XML::ExtOn - SAX Handler designed for funny work with XML. It | ||||||
139 | provides an easy-to-use interface for XML applications by adding objects. | ||||||
140 | |||||||
141 | XML::ExtOn override some SAX events. Each time an SAX event starts, | ||||||
142 | a method by that name prefixed with `on_' is called with the B<"blessed"> | ||||||
143 | Element object to be processed. | ||||||
144 | |||||||
145 | XML::ExtOn implement the following methods: | ||||||
146 | |||||||
147 | =over | ||||||
148 | |||||||
149 | =item * on_start_document | ||||||
150 | |||||||
151 | =item * on_start_prefix_mapping | ||||||
152 | |||||||
153 | =item * on_start_element | ||||||
154 | |||||||
155 | =item * on_end_element | ||||||
156 | |||||||
157 | =item * on_characters | ||||||
158 | |||||||
159 | =item * on_cdata | ||||||
160 | |||||||
161 | =back | ||||||
162 | |||||||
163 | XML::ExtOn put all B |
||||||
164 | |||||||
165 | It compliant XML namespaces (http://www.w3.org/TR/REC-xml-names/), by support | ||||||
166 | I |
||||||
167 | |||||||
168 | XML::ExtOn provide methods for create XML, such as C |
||||||
169 | |||||||
170 | =head1 FUNCTIONS | ||||||
171 | |||||||
172 | =cut | ||||||
173 | |||||||
174 | 6 | 6 | 239503 | use strict; | |||
6 | 18 | ||||||
6 | 290 | ||||||
175 | 6 | 6 | 36 | use warnings; | |||
6 | 15 | ||||||
6 | 4151 | ||||||
176 | |||||||
177 | 6 | 6 | 46 | use Carp; | |||
6 | 10 | ||||||
6 | 453 | ||||||
178 | 6 | 6 | 35 | use Data::Dumper; | |||
6 | 10 | ||||||
6 | 243 | ||||||
179 | |||||||
180 | 6 | 6 | 15030 | use XML::SAX::Base; | |||
6 | 143837 | ||||||
6 | 220 | ||||||
181 | 6 | 6 | 4352 | use XML::ExtOn::Element; | |||
6 | 20 | ||||||
6 | 155 | ||||||
182 | 6 | 6 | 3219 | use XML::ExtOn::Context; | |||
6 | 19 | ||||||
6 | 199 | ||||||
183 | 6 | 6 | 3301 | use XML::ExtOn::IncXML; | |||
6 | 13 | ||||||
6 | 154 | ||||||
184 | 6 | 6 | 5118 | use XML::Filter::SAX1toSAX2; | |||
6 | 26977 | ||||||
6 | 184 | ||||||
185 | 6 | 6 | 3260 | use XML::ExtOn::SAX12ExtOn; | |||
6 | 17 | ||||||
6 | 166 | ||||||
186 | 6 | 6 | 2622 | use XML::Parser::PerlSAX; | |||
0 | |||||||
0 | |||||||
187 | use Test::More; | ||||||
188 | |||||||
189 | require Exporter; | ||||||
190 | *import = \&Exporter::import; | ||||||
191 | @XML::ExtOn::EXPORT_OK = qw( create_pipe split_pipe); | ||||||
192 | |||||||
193 | sub _get_end_handler { | ||||||
194 | my $flt = shift; | ||||||
195 | my $handler = $flt->get_handler(); | ||||||
196 | |||||||
197 | return $handler if UNIVERSAL::isa( $handler, 'XML::ExtOn::Writer' ); | ||||||
198 | return $handler if UNIVERSAL::isa( $handler, 'XML::SAX::Writer::XML' ); | ||||||
199 | return $flt unless UNIVERSAL::isa( $handler, 'XML::SAX::Base' ); | ||||||
200 | return &_get_end_handler($handler); | ||||||
201 | } | ||||||
202 | |||||||
203 | =head1 create_pipe "flt_n1",$some_handler, $out_handler | ||||||
204 | |||||||
205 | use last arg as handler for out. | ||||||
206 | |||||||
207 | return parser ref. | ||||||
208 | |||||||
209 | my $h1 = new MyHandler1::; | ||||||
210 | my $filter = create_pipe( 'MyHandler1', $h1 ); | ||||||
211 | $filter->parse(' TEST |
||||||
212 | #also create pipe of pipes | ||||||
213 | my $filter1 = create_pipe( 'MyHandler1', 'MyHandler2' ); | ||||||
214 | my $h1 = new MyHandler3::; | ||||||
215 | my $filter2 = create_pipe( $filter1, $h1); | ||||||
216 | |||||||
217 | =cut | ||||||
218 | |||||||
219 | sub create_pipe { | ||||||
220 | |||||||
221 | my @args = reverse @_; | ||||||
222 | |||||||
223 | my $out_handler; | ||||||
224 | foreach my $f (@args) { | ||||||
225 | unless ( ref($f) ) { | ||||||
226 | unless ($out_handler) { | ||||||
227 | $out_handler = $f->new(); | ||||||
228 | } | ||||||
229 | else { | ||||||
230 | $out_handler = $f->new( Handler => $out_handler ); | ||||||
231 | } | ||||||
232 | } | ||||||
233 | elsif ( UNIVERSAL::isa( $f, 'XML::SAX::Base' ) ) { | ||||||
234 | unless ($out_handler) { | ||||||
235 | $out_handler = $f; | ||||||
236 | } | ||||||
237 | else { | ||||||
238 | my $end_handler = &_get_end_handler($f); | ||||||
239 | $end_handler->set_handler($out_handler); | ||||||
240 | $out_handler = $f; | ||||||
241 | } | ||||||
242 | } | ||||||
243 | else { | ||||||
244 | die "$f not SAX Drv"; | ||||||
245 | } | ||||||
246 | } | ||||||
247 | return $out_handler; | ||||||
248 | } | ||||||
249 | |||||||
250 | =head1 split_pipe $filter | ||||||
251 | |||||||
252 | return ref to array of filters in pipe | ||||||
253 | |||||||
254 | |||||||
255 | use XML::ExtOn qw(split_pipe create_pipe); | ||||||
256 | my $filter = create_pipe( 'MyHandler1', 'MyHandler2','MyHandler3'); | ||||||
257 | my $ref = @{ split_pipe( $filter) } [-1]; | ||||||
258 | isa_ok $ref, 'MyHandler3', 'check last element'; | ||||||
259 | |||||||
260 | =cut | ||||||
261 | |||||||
262 | sub split_pipe { | ||||||
263 | my $filter = shift || return []; | ||||||
264 | my @res = ($filter); | ||||||
265 | |||||||
266 | # use SAXed variable see XML::SAX::Base::get_handler() | ||||||
267 | if ( my $next = $filter->{Handler} ) { | ||||||
268 | #skip special SAX handlers | ||||||
269 | unless ( UNIVERSAL::isa( $next, 'XML::SAX::Base::NoHandler' ) ) { | ||||||
270 | push @res, @{ split_pipe($next) }; | ||||||
271 | } | ||||||
272 | } | ||||||
273 | return \@res; | ||||||
274 | } | ||||||
275 | |||||||
276 | use base 'XML::SAX::Base'; | ||||||
277 | use vars qw( $AUTOLOAD); | ||||||
278 | $XML::ExtOn::VERSION = '0.17'; | ||||||
279 | ### install get/set accessors for this object. | ||||||
280 | for my $key ( | ||||||
281 | qw/ context _objects_stack _cdata_mode _cdata_characters _root_stack /) | ||||||
282 | { | ||||||
283 | no strict 'refs'; | ||||||
284 | *{ __PACKAGE__ . "::$key" } = sub { | ||||||
285 | my $self = shift; | ||||||
286 | $self->{___EXT_on_attrs}->{$key} = $_[0] if @_; | ||||||
287 | return $self->{___EXT_on_attrs}->{$key}; | ||||||
288 | } | ||||||
289 | } | ||||||
290 | |||||||
291 | =head1 METHODS | ||||||
292 | |||||||
293 | =cut | ||||||
294 | |||||||
295 | sub new { | ||||||
296 | my $class = shift; | ||||||
297 | my $self = &XML::SAX::Base::new( $class, @_, ); | ||||||
298 | $self->_objects_stack( [] ); | ||||||
299 | $self->_root_stack( [] ); #init incoming stack of start end | ||||||
300 | $self->_cdata_mode(0); | ||||||
301 | my $buf; | ||||||
302 | $self->_cdata_characters( \$buf ); #setup cdata buffer | ||||||
303 | my $doc_context = new XML::ExtOn::Context::; | ||||||
304 | $self->context($doc_context); | ||||||
305 | return $self; | ||||||
306 | } | ||||||
307 | |||||||
308 | =head2 on_start_document $document | ||||||
309 | |||||||
310 | Method handle C |
||||||
311 | variables. | ||||||
312 | |||||||
313 | sub on_start_document { | ||||||
314 | my $self = shift; | ||||||
315 | $self->{_LINKS_ARRAY} = []; | ||||||
316 | $self->SUPER::on_start_document(@_); | ||||||
317 | } | ||||||
318 | |||||||
319 | =cut | ||||||
320 | |||||||
321 | sub on_start_document { | ||||||
322 | my ( $self, $document ) = @_; | ||||||
323 | $self->SUPER::start_document($document); | ||||||
324 | } | ||||||
325 | |||||||
326 | sub start_document { | ||||||
327 | my ( $self, $document ) = @_; | ||||||
328 | return if $self->{___EXT_on_attrs}->{_skip_start_docs}++; | ||||||
329 | $self->on_start_document($document); | ||||||
330 | } | ||||||
331 | |||||||
332 | sub end_document { | ||||||
333 | my $self = shift; | ||||||
334 | my $var = --$self->{___EXT_on_attrs}->{_skip_start_docs}; | ||||||
335 | return if $var; | ||||||
336 | $self->SUPER::end_document(@_); | ||||||
337 | } | ||||||
338 | |||||||
339 | =head2 on_start_prefix_mapping prefix1=>ns_uri1[, prefix2=>ns_uri2] | ||||||
340 | |||||||
341 | Called on C |
||||||
342 | |||||||
343 | sub on_start_prefix_mapping { | ||||||
344 | my $self = shift; | ||||||
345 | my %map = @_; | ||||||
346 | $self->SUPER::start_prefix_mapping(@_) | ||||||
347 | } | ||||||
348 | |||||||
349 | =cut | ||||||
350 | |||||||
351 | sub on_start_prefix_mapping { | ||||||
352 | my $self = shift; | ||||||
353 | my %map = @_; | ||||||
354 | while ( my ( $pref, $ns_uri ) = each %map ) { | ||||||
355 | $self->add_namespace( $pref, $ns_uri ); | ||||||
356 | $self->SUPER::start_prefix_mapping( | ||||||
357 | { | ||||||
358 | Prefix => $pref, | ||||||
359 | NamespaceURI => $ns_uri | ||||||
360 | } | ||||||
361 | ); | ||||||
362 | } | ||||||
363 | } | ||||||
364 | |||||||
365 | # | ||||||
366 | # { Prefix => 'xlink', NamespaceURI => 'http://www.w3.org/1999/xlink' } | ||||||
367 | # | ||||||
368 | |||||||
369 | sub start_prefix_mapping { | ||||||
370 | my $self = shift; | ||||||
371 | |||||||
372 | #declare namespace for current context | ||||||
373 | my %map = (); | ||||||
374 | foreach my $ref (@_) { | ||||||
375 | my ( $prefix, $ns_uri ) = @{$ref}{qw/Prefix NamespaceURI/}; | ||||||
376 | $map{$prefix} = $ns_uri; | ||||||
377 | } | ||||||
378 | $self->on_start_prefix_mapping(%map); | ||||||
379 | } | ||||||
380 | |||||||
381 | =head2 on_start_element $elem | ||||||
382 | |||||||
383 | Method handle C |
||||||
384 | |||||||
385 | Method must return C<$elem> or ref to array of objects. | ||||||
386 | |||||||
387 | For example: | ||||||
388 | |||||||
389 | sub on_start_element { | ||||||
390 | my $self = shift; | ||||||
391 | my $elem = shift; | ||||||
392 | $elem->add_content( $self->mk_cdata("test")); | ||||||
393 | return $elem | ||||||
394 | } | ||||||
395 | ... | ||||||
396 | |||||||
397 | return [ $elem, ,$self->mk_element("after_start_elem") ] | ||||||
398 | |||||||
399 | return [ $self->mk_element("before_start_elem"), $elem ] | ||||||
400 | ... | ||||||
401 | |||||||
402 | =cut | ||||||
403 | |||||||
404 | sub on_start_element { | ||||||
405 | shift; | ||||||
406 | return [@_]; | ||||||
407 | } | ||||||
408 | |||||||
409 | sub __expand_on_start { | ||||||
410 | my $self = shift; | ||||||
411 | my $obj = shift || return []; | ||||||
412 | # warn "before _expand $obj".Dumper($obj) if $obj->local_name eq 'feed'; | ||||||
413 | my $res = $self->on_start_element($obj); | ||||||
414 | # warn "_expand $obj".Dumper($res , $obj) if $obj->local_name eq 'feed'; | ||||||
415 | my @stack = | ||||||
416 | $res | ||||||
417 | ? ref($res) eq 'ARRAY' | ||||||
418 | ? @{$res} | ||||||
419 | : ($res) | ||||||
420 | : (); | ||||||
421 | |||||||
422 | #add self object | ||||||
423 | push @stack, $obj; | ||||||
424 | |||||||
425 | #expand wrap_around and insert_to | ||||||
426 | # also remove dups for $obj | ||||||
427 | my %uniq = (); | ||||||
428 | my @res = (); | ||||||
429 | foreach my $o (@stack) { | ||||||
430 | |||||||
431 | # also remove dups for $obj | ||||||
432 | next if $uniq{$o}++; | ||||||
433 | unless ( UNIVERSAL::isa( $o, 'XML::ExtOn::Element' ) ) { | ||||||
434 | |||||||
435 | #don'n touch any events | ||||||
436 | push @res, $o; | ||||||
437 | } | ||||||
438 | else { | ||||||
439 | |||||||
440 | #convert any object to events (exept $obj) | ||||||
441 | unless ( $o eq $obj ) { | ||||||
442 | push @res, $self->mk_start_element($o), | ||||||
443 | $self->mk_process_stack($o), $self->mk_end_element($o); | ||||||
444 | } | ||||||
445 | else { | ||||||
446 | |||||||
447 | #expand $insert_to | ||||||
448 | my $insert_to = $o->_wrap_begin || []; | ||||||
449 | if ( scalar @{$insert_to} ) { | ||||||
450 | for ( @{$insert_to} ) { | ||||||
451 | push @res, $self->mk_start_element($_); | ||||||
452 | } | ||||||
453 | } | ||||||
454 | |||||||
455 | # $o->_wrap_begin([]); | ||||||
456 | #insert result event to write tag | ||||||
457 | push @res, $self->_mk_event_start_element($o); | ||||||
458 | |||||||
459 | #process elemet's stack (add_content) | ||||||
460 | push @res, $self->mk_process_stack($o); | ||||||
461 | |||||||
462 | #ad wrap_around started | ||||||
463 | my $waround = $o->_wrap_around_start || []; | ||||||
464 | if ( scalar @{$waround} ) { | ||||||
465 | for ( @{$waround} ) { | ||||||
466 | push @res, $self->mk_start_element($_); | ||||||
467 | } | ||||||
468 | } | ||||||
469 | |||||||
470 | # $o->_wrap_around_start([]); | ||||||
471 | } | ||||||
472 | } | ||||||
473 | } | ||||||
474 | |||||||
475 | #now expand | ||||||
476 | return \@res; | ||||||
477 | } | ||||||
478 | |||||||
479 | sub start_element { | ||||||
480 | my $self = shift; | ||||||
481 | my $current_obj = shift; | ||||||
482 | |||||||
483 | die "empty" . Dumper( [ map { [ caller($_) ] } ( 0 .. 4 ) ] ) | ||||||
484 | unless defined $current_obj; | ||||||
485 | |||||||
486 | unless ( UNIVERSAL::isa( $current_obj, 'XML::ExtOn::Element' ) ) { | ||||||
487 | my $context; | ||||||
488 | if ( my $current_root_element = $self->current_root_element ) { | ||||||
489 | $context = $current_root_element->ns->sub_context(); | ||||||
490 | } | ||||||
491 | $current_obj = | ||||||
492 | $self->__mk_element_from_sax2( $current_obj, context => $context ); | ||||||
493 | } | ||||||
494 | else { | ||||||
495 | |||||||
496 | #set new context | ||||||
497 | my $new_context; | ||||||
498 | if ( my $current_root_element = $self->current_root_element ) { | ||||||
499 | $new_context = $current_root_element->ns->sub_context(); | ||||||
500 | } | ||||||
501 | $new_context ||= $self->context->sub_context(); | ||||||
502 | #save changes (for namespaces) | ||||||
503 | my $changes = $current_obj->ns->get_changes(); | ||||||
504 | while (my ($prefix, $val) = each %$changes) { | ||||||
505 | $new_context->declare_prefix($prefix, $val); | ||||||
506 | } | ||||||
507 | $current_obj->_context($new_context); | ||||||
508 | } | ||||||
509 | |||||||
510 | my $current_root_element = $self->current_root_element; | ||||||
511 | |||||||
512 | #push to stack of incoming objects | ||||||
513 | push @{ $self->_root_stack() }, $current_obj; | ||||||
514 | |||||||
515 | #=comment check skip | ||||||
516 | #check current root element for skip_content | ||||||
517 | if ($current_root_element) { | ||||||
518 | my $skip_content = $current_root_element->is_skip_content; | ||||||
519 | if ($skip_content) { | ||||||
520 | $current_root_element->is_skip_content( ++$skip_content ); | ||||||
521 | return; | ||||||
522 | } | ||||||
523 | } | ||||||
524 | |||||||
525 | #=cut | ||||||
526 | #warn ref($self).":START for " . $current_obj->local_name; | ||||||
527 | return $self->__start_element($current_obj); | ||||||
528 | } | ||||||
529 | |||||||
530 | sub __start_element { | ||||||
531 | my $self = shift; | ||||||
532 | my $current_obj = shift; | ||||||
533 | |||||||
534 | #check current element for skip_content | ||||||
535 | if ( my $current_element = $self->current_element ) { | ||||||
536 | my $skip_content = $current_element->is_skip_content; | ||||||
537 | if ( $skip_content > 1 ) { | ||||||
538 | $current_element->is_skip_content( --$skip_content ); | ||||||
539 | return; | ||||||
540 | } | ||||||
541 | } | ||||||
542 | |||||||
543 | #call __start_element | ||||||
544 | my $res = $self->__expand_on_start($current_obj); | ||||||
545 | $current_obj->{_expanded_on_start} = scalar(@$res); | ||||||
546 | # warn ref($self) . "start_exp: " . $current_obj->local_name . ": " . Dumper( | ||||||
547 | # [ | ||||||
548 | # map { | ||||||
549 | # ref($_) eq 'HASH' | ||||||
550 | # ? $_->{type} . ":" . $_->{data}->local_name | ||||||
551 | # : $_->local_name | ||||||
552 | # } @$res | ||||||
553 | # ] | ||||||
554 | # ); | ||||||
555 | |||||||
556 | #walk via array | ||||||
557 | foreach my $elem (@$res) { | ||||||
558 | |||||||
559 | unless ( UNIVERSAL::isa( $elem, 'XML::ExtOn::Element' ) ) { | ||||||
560 | |||||||
561 | #run event | ||||||
562 | #warn $elem->{type}; | ||||||
563 | $self->_process_comm($elem); | ||||||
564 | } | ||||||
565 | else { | ||||||
566 | |||||||
567 | #register new namespaces | ||||||
568 | my $changes = $current_obj->ns->get_changes; | ||||||
569 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
570 | |||||||
571 | for ( keys %$changes ) { | ||||||
572 | $self->end_prefix_mapping( | ||||||
573 | { | ||||||
574 | Prefix => $_, | ||||||
575 | NamespaceURI => $parent_map->{$_}, | ||||||
576 | } | ||||||
577 | ) if exists $parent_map->{$_}; | ||||||
578 | $self->start_prefix_mapping( | ||||||
579 | { | ||||||
580 | Prefix => $_, | ||||||
581 | NamespaceURI => $changes->{$_}, | ||||||
582 | } | ||||||
583 | ); | ||||||
584 | } | ||||||
585 | |||||||
586 | #save element in stack | ||||||
587 | push @{ $self->_objects_stack() }, $current_obj; | ||||||
588 | my @object_stack = @{ $current_obj->_stack }; | ||||||
589 | $current_obj->_stack( [] ); | ||||||
590 | |||||||
591 | #skip deleted elements from xml stream | ||||||
592 | unless ( $current_obj->is_delete_element ) { | ||||||
593 | |||||||
594 | # warn "$self: process start ".$current_obj->local_name; | ||||||
595 | if ( UNIVERSAL::isa( $self->{Handler}, 'XML::ExtOn' ) ) { | ||||||
596 | my $cloned = $current_obj->__clone; | ||||||
597 | unless ( $self->{__make_self_events} ) { | ||||||
598 | $self->{Handler}->start_element($cloned); | ||||||
599 | } | ||||||
600 | else { | ||||||
601 | $self->{Handler}->__start_element($cloned); | ||||||
602 | |||||||
603 | } | ||||||
604 | } | ||||||
605 | else { | ||||||
606 | my $res_data = $self->__exp_element_to_sax2($current_obj); | ||||||
607 | $self->SUPER::start_element($res_data); | ||||||
608 | } | ||||||
609 | } | ||||||
610 | unless ( $current_obj->is_skip_content ) { | ||||||
611 | $self->_process_comm($_) for @object_stack; | ||||||
612 | } | ||||||
613 | } | ||||||
614 | |||||||
615 | } | ||||||
616 | } | ||||||
617 | |||||||
618 | =head2 on_end_element $elem | ||||||
619 | |||||||
620 | Method handle C |
||||||
621 | It call before end if element. | ||||||
622 | |||||||
623 | Method must return C<$elem> or ref to array of objects. | ||||||
624 | |||||||
625 | For example: | ||||||
626 | |||||||
627 | sub on_end_element { | ||||||
628 | my $self = shift; | ||||||
629 | my $elem = shift; | ||||||
630 | if ( $elem->is_delete_element ) { | ||||||
631 | warn $elem->local_name . " deleted"; | ||||||
632 | return [ $elem, $self->mk_element("after_deleted_elem") ] | ||||||
633 | }; | ||||||
634 | return $elem | ||||||
635 | } | ||||||
636 | ... | ||||||
637 | |||||||
638 | return [ $elem, ,$self->mk_element("after_close_tag_of_elem") ] | ||||||
639 | |||||||
640 | return [ $self->mk_element("before_close_tag_of_elem"), $elem ] | ||||||
641 | ... | ||||||
642 | |||||||
643 | =cut | ||||||
644 | |||||||
645 | sub on_end_element { | ||||||
646 | shift; | ||||||
647 | return [@_]; | ||||||
648 | } | ||||||
649 | |||||||
650 | sub __expand_on_end { | ||||||
651 | my $self = shift; | ||||||
652 | my $obj = shift || return []; | ||||||
653 | |||||||
654 | # | ||||||
655 | my $res = $self->on_end_element($obj); | ||||||
656 | my @stack = | ||||||
657 | $res | ||||||
658 | ? ref($res) eq 'ARRAY' | ||||||
659 | ? @{$res} | ||||||
660 | : ($res) | ||||||
661 | : (); | ||||||
662 | |||||||
663 | #add self object | ||||||
664 | push @stack, $obj; | ||||||
665 | |||||||
666 | #expand wrap_around and insert_to | ||||||
667 | # also remove dups for $obj | ||||||
668 | my %uniq = (); | ||||||
669 | my @res = (); | ||||||
670 | foreach my $o (@stack) { | ||||||
671 | |||||||
672 | # also remove dups for $obj | ||||||
673 | next if $uniq{$o}++; | ||||||
674 | unless ( UNIVERSAL::isa( $o, 'XML::ExtOn::Element' ) ) { | ||||||
675 | |||||||
676 | #don'n touch any events | ||||||
677 | push @res, $o; | ||||||
678 | } | ||||||
679 | else { | ||||||
680 | |||||||
681 | #convert any object to events (exept $obj) | ||||||
682 | unless ( $o eq $obj ) { | ||||||
683 | push @res, $self->mk_start_element($o), | ||||||
684 | $self->mk_process_stack($o), $self->mk_end_element($o); | ||||||
685 | } | ||||||
686 | else { | ||||||
687 | |||||||
688 | #ad wrap_around started | ||||||
689 | my $waround = $o->_wrap_around_end || []; | ||||||
690 | if ( scalar @{$waround} ) { | ||||||
691 | for ( reverse @{$waround} ) { | ||||||
692 | push @res, $self->mk_end_element($_); | ||||||
693 | } | ||||||
694 | } | ||||||
695 | |||||||
696 | # push @res, $o; #add object | ||||||
697 | #process elemet's stack (add_content) | ||||||
698 | push @res, $self->mk_process_stack($o); | ||||||
699 | |||||||
700 | #expand $insert_to | ||||||
701 | push @res, $self->_mk_event_end_element($o); | ||||||
702 | |||||||
703 | my $insert_to = $o->_wrap_end || []; | ||||||
704 | if ( scalar @{$insert_to} ) { | ||||||
705 | for ( reverse @{$insert_to} ) { | ||||||
706 | push @res, $self->mk_end_element($_); | ||||||
707 | } | ||||||
708 | } | ||||||
709 | } | ||||||
710 | } | ||||||
711 | } | ||||||
712 | |||||||
713 | #now expand | ||||||
714 | return \@res; | ||||||
715 | } | ||||||
716 | |||||||
717 | sub end_element { | ||||||
718 | my $self = shift; | ||||||
719 | my $data = shift; | ||||||
720 | |||||||
721 | #get current element | ||||||
722 | #pop from stack of incoming objects | ||||||
723 | $data = pop @{ $self->_root_stack() }; | ||||||
724 | die " $self empty stack" . Dumper( [ map { [ caller($_) ] } ( 0 .. 4 ) ] ) | ||||||
725 | unless defined $data; | ||||||
726 | |||||||
727 | # warn "do __end; for " | ||||||
728 | # . $data->local_name | ||||||
729 | # . " {_expanded_on_start}" | ||||||
730 | # . $data->{_expanded_on_start}; | ||||||
731 | |||||||
732 | #check current element for skip_content | ||||||
733 | if ( my $current_root_element = $self->current_root_element ) { | ||||||
734 | my $skip_content = $current_root_element->is_skip_content; | ||||||
735 | if ( $skip_content > 1 ) { | ||||||
736 | $current_root_element->is_skip_content( --$skip_content ); | ||||||
737 | return; | ||||||
738 | } | ||||||
739 | } | ||||||
740 | |||||||
741 | # warn ref($self).":END for " . $data->local_name; | ||||||
742 | # if ( my $started = $data->{_expanded_on_start} ) { | ||||||
743 | # for ( 1..$started-1 ) { | ||||||
744 | # $self->__end_element($data); | ||||||
745 | # } | ||||||
746 | # } | ||||||
747 | return $self->__end_element($data); | ||||||
748 | } | ||||||
749 | |||||||
750 | sub __end_element { | ||||||
751 | my $self = shift; | ||||||
752 | |||||||
753 | my $current_obj = shift; #may be use for control stack | ||||||
754 | #pop element from stack | ||||||
755 | |||||||
756 | # my $current_obj1 = pop @{ $self->_objects_stack() }; | ||||||
757 | |||||||
758 | my $res = $self->__expand_on_end($current_obj); | ||||||
759 | |||||||
760 | # warn ref($self)."end_exp: " | ||||||
761 | # . $current_obj->local_name . ": " | ||||||
762 | # . Dumper( | ||||||
763 | # [ | ||||||
764 | # map { ref($_) eq 'HASH' ? $_->{type}.":".$_->{data}->local_name : $_->local_name } | ||||||
765 | # @$res | ||||||
766 | # ] | ||||||
767 | # ); | ||||||
768 | |||||||
769 | foreach my $elem (@$res) { | ||||||
770 | unless ( UNIVERSAL::isa( $elem, 'XML::ExtOn::Element' ) ) { | ||||||
771 | |||||||
772 | #run event | ||||||
773 | $self->_process_comm($elem); | ||||||
774 | } | ||||||
775 | else { | ||||||
776 | die "END!!"; | ||||||
777 | |||||||
778 | #setup default ns | ||||||
779 | my $data = $current_obj->to_sax2; | ||||||
780 | delete $data->{Attributes}; | ||||||
781 | $data->{NamespaceURI} = $current_obj->default_uri; | ||||||
782 | |||||||
783 | # if skip | ||||||
784 | #check current element for skip_content | ||||||
785 | if ( my $current_element = $self->current_element ) { | ||||||
786 | my $skip_content = $current_element->is_skip_content; | ||||||
787 | if ( $skip_content > 1 ) { | ||||||
788 | $current_element->is_skip_content( --$skip_content ); | ||||||
789 | return; | ||||||
790 | } | ||||||
791 | } | ||||||
792 | |||||||
793 | unless ( $current_obj->is_skip_content ) { | ||||||
794 | $self->_process_comm($_) for @{ $current_obj->_stack }; | ||||||
795 | $current_obj->_stack( [] ); | ||||||
796 | } | ||||||
797 | |||||||
798 | unless ( $current_obj->is_delete_element ) { | ||||||
799 | |||||||
800 | # warn "$self: process end ".$current_obj->local_name; | ||||||
801 | unless ( $self->{__make_self_events} ) { | ||||||
802 | $self->SUPER::end_element($data); | ||||||
803 | } | ||||||
804 | else { | ||||||
805 | $self->{Handler}->__end_element($data); | ||||||
806 | } | ||||||
807 | } | ||||||
808 | |||||||
809 | my $changes = $current_obj->ns->get_changes; | ||||||
810 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
811 | for ( keys %$changes ) { | ||||||
812 | $self->end_prefix_mapping( | ||||||
813 | { | ||||||
814 | Prefix => $_, | ||||||
815 | NamespaceURI => $changes->{$_}, | ||||||
816 | } | ||||||
817 | ); | ||||||
818 | if ( exists( $parent_map->{$_} ) ) { | ||||||
819 | $self->start_prefix_mapping( | ||||||
820 | { | ||||||
821 | Prefix => $_, | ||||||
822 | NamespaceURI => $parent_map->{$_}, | ||||||
823 | } | ||||||
824 | ); | ||||||
825 | } | ||||||
826 | } | ||||||
827 | } | ||||||
828 | } | ||||||
829 | } | ||||||
830 | |||||||
831 | =head2 on_characters( $self->current_element, $data->{Data} ) | ||||||
832 | |||||||
833 | Must return string for write to stream. | ||||||
834 | |||||||
835 | sub on_characters { | ||||||
836 | my ( $self, $elem, $str ) = @_; | ||||||
837 | #lowercase all characters | ||||||
838 | return lc $str; | ||||||
839 | } | ||||||
840 | |||||||
841 | |||||||
842 | =cut | ||||||
843 | |||||||
844 | sub on_characters { | ||||||
845 | my ( $self, $elem, $str ) = @_; | ||||||
846 | return $str; | ||||||
847 | } | ||||||
848 | |||||||
849 | =head2 on_cdata ( $current_element, $data ) | ||||||
850 | |||||||
851 | Must return string for write to stream | ||||||
852 | |||||||
853 | sub on_cdata { | ||||||
854 | my ( $self, $elem, $str ) = @_; | ||||||
855 | return lc $str; | ||||||
856 | } | ||||||
857 | |||||||
858 | =cut | ||||||
859 | |||||||
860 | sub on_cdata { | ||||||
861 | my ( $self, $elem, $str ) = @_; | ||||||
862 | return $str; | ||||||
863 | } | ||||||
864 | |||||||
865 | #set flag for cdata content | ||||||
866 | |||||||
867 | sub start_cdata { | ||||||
868 | my $self = shift; | ||||||
869 | $self->_cdata_mode(1); | ||||||
870 | return; | ||||||
871 | } | ||||||
872 | |||||||
873 | #set flag to end cdata | ||||||
874 | |||||||
875 | sub end_cdata { | ||||||
876 | my $self = shift; | ||||||
877 | if ( my $elem = $self->current_element | ||||||
878 | and defined( my $cdata_buf = ${ $self->_cdata_characters } ) ) | ||||||
879 | { | ||||||
880 | |||||||
881 | if ( defined( my $data = $self->on_cdata( $elem, $cdata_buf ) ) ) { | ||||||
882 | $self->SUPER::start_cdata; | ||||||
883 | $self->SUPER::characters( { Data => $data } ); | ||||||
884 | $self->SUPER::end_cdata; | ||||||
885 | } | ||||||
886 | } | ||||||
887 | |||||||
888 | #after all clear cd_data_buffer and reset cd_data mode flag | ||||||
889 | my $new_buf; | ||||||
890 | $self->_cdata_characters( \$new_buf ); | ||||||
891 | $self->_cdata_mode(0); | ||||||
892 | return; | ||||||
893 | } | ||||||
894 | |||||||
895 | sub characters { | ||||||
896 | my $self = shift; | ||||||
897 | my ($data) = @_; | ||||||
898 | |||||||
899 | # warn "$self do chars" . $data->{Data}; | ||||||
900 | |||||||
901 | #skip childs elements characters ( > 1 ) and self text ( > 0) | ||||||
902 | if ( $self->current_element ) { | ||||||
903 | return if $self->current_element->is_skip_content; | ||||||
904 | } | ||||||
905 | else { | ||||||
906 | |||||||
907 | #skip characters without element | ||||||
908 | return; | ||||||
909 | } | ||||||
910 | |||||||
911 | #for cdata section collect characters in buffer | ||||||
912 | if ( $self->_cdata_mode ) { | ||||||
913 | |||||||
914 | # warn "$self do CDATA" . $data->{Data}; | ||||||
915 | # warn " $self CDTATA" . Dumper( [ map { [ caller($_) ] } ( 0 .. 10 ) ] ); | ||||||
916 | # unless defined $data; | ||||||
917 | |||||||
918 | ${ $self->_cdata_characters } .= $data->{Data}; | ||||||
919 | return; | ||||||
920 | } | ||||||
921 | |||||||
922 | #collect chars fo current element | ||||||
923 | if ( | ||||||
924 | defined( | ||||||
925 | my $str = | ||||||
926 | $self->on_characters( $self->current_element, $data->{Data} ) | ||||||
927 | ) | ||||||
928 | ) | ||||||
929 | { | ||||||
930 | return $self->SUPER::characters( { Data => $str } ); | ||||||
931 | } | ||||||
932 | } | ||||||
933 | |||||||
934 | =head2 mk_element |
||||||
935 | |||||||
936 | Return object of element item for include to stream. | ||||||
937 | |||||||
938 | =cut | ||||||
939 | |||||||
940 | sub mk_element { | ||||||
941 | my $self = shift; | ||||||
942 | my $name = shift; | ||||||
943 | my %args = @_; | ||||||
944 | if ( my $current_element = $self->current_element ) { | ||||||
945 | $args{context} = $current_element->ns->sub_context(); | ||||||
946 | } | ||||||
947 | $args{context} ||= $self->context->sub_context(); | ||||||
948 | my $elem = new XML::ExtOn::Element:: | ||||||
949 | name => $name, | ||||||
950 | %args; | ||||||
951 | return $elem; | ||||||
952 | } | ||||||
953 | |||||||
954 | =head2 mk_from_xml |
||||||
955 | |||||||
956 | Return command for include to stream. | ||||||
957 | |||||||
958 | =cut | ||||||
959 | |||||||
960 | sub mk_from_xml { | ||||||
961 | my $self = shift; | ||||||
962 | my $string = shift; | ||||||
963 | my $skip_tmp_root = | ||||||
964 | XML::ExtOn::IncXML->new( Handler => $self, __make_self_events => 1 ); | ||||||
965 | my $sax2_filter = XML::Filter::SAX1toSAX2->new( Handler => $skip_tmp_root ); | ||||||
966 | my $parser = XML::Parser::PerlSAX->new( | ||||||
967 | { | ||||||
968 | Handler => $sax2_filter, | ||||||
969 | Source => { String => " |
||||||
970 | } | ||||||
971 | ); | ||||||
972 | return $parser; | ||||||
973 | } | ||||||
974 | |||||||
975 | =head2 mk_cdata $string | \$string | ||||||
976 | |||||||
977 | return command for insert cdata to stream | ||||||
978 | |||||||
979 | =cut | ||||||
980 | |||||||
981 | sub mk_cdata { | ||||||
982 | my $self = shift; | ||||||
983 | my $string = shift; | ||||||
984 | return { type => 'CDATA', data => ref($string) ? $string : \$string }; | ||||||
985 | } | ||||||
986 | |||||||
987 | =head2 mk_characters $string | \$string | ||||||
988 | |||||||
989 | return command for insert characters to stream | ||||||
990 | |||||||
991 | =cut | ||||||
992 | |||||||
993 | sub mk_characters { | ||||||
994 | my $self = shift; | ||||||
995 | my $string = shift; | ||||||
996 | return { type => 'CHARACTERS', data => ref($string) ? $string : \$string }; | ||||||
997 | } | ||||||
998 | |||||||
999 | =head2 mk_start_element |
||||||
1000 | |||||||
1001 | return command for start element event | ||||||
1002 | |||||||
1003 | =cut | ||||||
1004 | |||||||
1005 | sub mk_start_element { | ||||||
1006 | my $self = shift; | ||||||
1007 | my $elem = shift; | ||||||
1008 | return { type => 'START_ELEMENT', data => $elem }; | ||||||
1009 | } | ||||||
1010 | |||||||
1011 | =head2 mk_event_element |
||||||
1012 | |||||||
1013 | return command for expand stack for element | ||||||
1014 | |||||||
1015 | =cut | ||||||
1016 | |||||||
1017 | sub mk_process_stack { | ||||||
1018 | my $self = shift; | ||||||
1019 | my $elem = shift; | ||||||
1020 | my @objects = @{ $elem->_stack }; | ||||||
1021 | $elem->_stack( [] ); | ||||||
1022 | return { type => 'STACK', data => $elem, objects => \@objects }; | ||||||
1023 | } | ||||||
1024 | |||||||
1025 | =head2 _mk_event_start_element |
||||||
1026 | |||||||
1027 | return start tag command. (internal) | ||||||
1028 | |||||||
1029 | =cut | ||||||
1030 | |||||||
1031 | sub _mk_event_start_element { | ||||||
1032 | my $self = shift; | ||||||
1033 | my $elem = shift; | ||||||
1034 | return { type => 'EV_START_ELEMENT', data => $elem }; | ||||||
1035 | } | ||||||
1036 | |||||||
1037 | =head2 _mk_event_end_element |
||||||
1038 | |||||||
1039 | return end tag command. (internal) | ||||||
1040 | |||||||
1041 | =cut | ||||||
1042 | |||||||
1043 | sub _mk_event_end_element { | ||||||
1044 | my $self = shift; | ||||||
1045 | my $elem = shift; | ||||||
1046 | return { type => 'EV_END_ELEMENT', data => $elem }; | ||||||
1047 | } | ||||||
1048 | |||||||
1049 | =head2 mk_end_element |
||||||
1050 | |||||||
1051 | return command for end element event | ||||||
1052 | |||||||
1053 | =cut | ||||||
1054 | |||||||
1055 | sub mk_end_element { | ||||||
1056 | my $self = shift; | ||||||
1057 | my $elem = shift; | ||||||
1058 | return { type => 'END_ELEMENT', data => $elem }; | ||||||
1059 | } | ||||||
1060 | |||||||
1061 | sub __mk_element_from_sax2 { | ||||||
1062 | my $self = shift; | ||||||
1063 | my $data = shift; | ||||||
1064 | my $elem = $self->mk_element( $data->{LocalName}, sax2 => $data, @_ ); | ||||||
1065 | return $elem; | ||||||
1066 | } | ||||||
1067 | |||||||
1068 | sub __exp_element_to_sax2 { | ||||||
1069 | my $self = shift; | ||||||
1070 | my $elem = shift; | ||||||
1071 | return $elem->to_sax2; | ||||||
1072 | } | ||||||
1073 | |||||||
1074 | =head2 current_element | ||||||
1075 | |||||||
1076 | Return link to current processing element. | ||||||
1077 | |||||||
1078 | =cut | ||||||
1079 | |||||||
1080 | sub current_element { | ||||||
1081 | my $self = shift; | ||||||
1082 | if ( my $stack = $self->_objects_stack() ) { | ||||||
1083 | return $stack->[-1]; | ||||||
1084 | } | ||||||
1085 | return; | ||||||
1086 | } | ||||||
1087 | |||||||
1088 | =head2 current_root_element | ||||||
1089 | |||||||
1090 | Return link to current root element in incoming stack. | ||||||
1091 | Used in start_element and end_element methods | ||||||
1092 | |||||||
1093 | =cut | ||||||
1094 | |||||||
1095 | sub current_root_element { | ||||||
1096 | my $self = shift; | ||||||
1097 | if ( my $stack = $self->_root_stack() ) { | ||||||
1098 | return $stack->[-1]; | ||||||
1099 | } | ||||||
1100 | return; | ||||||
1101 | } | ||||||
1102 | |||||||
1103 | # Private method for process commands | ||||||
1104 | |||||||
1105 | sub _process_comm { | ||||||
1106 | my $self = shift; | ||||||
1107 | my $comm = shift || return; | ||||||
1108 | if ( UNIVERSAL::isa( $comm, 'XML::Parser::PerlSAX' ) ) { | ||||||
1109 | $comm->parse(); | ||||||
1110 | } | ||||||
1111 | elsif ( UNIVERSAL::isa( $comm, 'XML::Parser' ) ) { | ||||||
1112 | warn "parser!"; | ||||||
1113 | $comm->parse(); | ||||||
1114 | } | ||||||
1115 | elsif ( UNIVERSAL::isa( $comm, 'XML::ExtOn::Element' ) ) { | ||||||
1116 | |||||||
1117 | # warn ref($self)."start ELEMENT " . $comm->local_name; | ||||||
1118 | $self->__start_element($comm); | ||||||
1119 | |||||||
1120 | # while ( my $obj = shift @{ $comm->_stack } ) { | ||||||
1121 | # $self->_process_comm($obj); | ||||||
1122 | # } | ||||||
1123 | $self->__end_element($comm); | ||||||
1124 | |||||||
1125 | # warn ref($self)."end ELEMENT " . $comm->local_name; | ||||||
1126 | ; # unless shift; #if exists extra param not end elem | ||||||
1127 | } | ||||||
1128 | elsif ( ref($comm) eq 'HASH' and exists $comm->{type} ) { | ||||||
1129 | if ( $comm->{type} eq 'CDATA' ) { | ||||||
1130 | |||||||
1131 | #warn "$self : DO CDATA!!!"; | ||||||
1132 | $self->start_cdata; | ||||||
1133 | $self->characters( { Data => ${ $comm->{data} } } ); | ||||||
1134 | $self->end_cdata; | ||||||
1135 | } | ||||||
1136 | elsif ( $comm->{type} eq 'CHARACTERS' ) { | ||||||
1137 | unless ( ref( $comm->{data} ) eq 'SCALAR' ) { | ||||||
1138 | warn "NOT REF" . Dumper $comm; | ||||||
1139 | warn "empty" . Dumper( [ map { [ caller($_) ] } ( 0 .. 16 ) ] ); | ||||||
1140 | exit; | ||||||
1141 | |||||||
1142 | } | ||||||
1143 | $self->characters( { Data => ${ $comm->{data} } } ); | ||||||
1144 | } | ||||||
1145 | elsif ( $comm->{type} eq 'START_ELEMENT' ) { | ||||||
1146 | my $current_obj = $comm->{data}; | ||||||
1147 | $self->__start_element( $comm->{data} ); | ||||||
1148 | } | ||||||
1149 | elsif ( $comm->{type} eq 'END_ELEMENT' ) { | ||||||
1150 | my $current_obj = $comm->{data}; | ||||||
1151 | $self->__end_element( $comm->{data} ); | ||||||
1152 | } | ||||||
1153 | elsif ( $comm->{type} eq 'STACK' ) { | ||||||
1154 | my $stack = $comm->{objects}; | ||||||
1155 | my $comm = $comm->{data}; | ||||||
1156 | |||||||
1157 | # warn "$self: ", | ||||||
1158 | # $comm->local_name . " stack: " . scalar( @{$stack} ) . Dumper( | ||||||
1159 | # [ | ||||||
1160 | # map { | ||||||
1161 | # ref($_) eq 'HASH' | ||||||
1162 | # ? $_->{type} . ":" . '$_->{data}->local_name' | ||||||
1163 | # : $_->local_name | ||||||
1164 | # } @$stack | ||||||
1165 | # ] | ||||||
1166 | # ); | ||||||
1167 | # warn ref($self)."START PROCESS STACK ".$comm->local_name; | ||||||
1168 | while ( my $obj = shift @{$stack} ) { | ||||||
1169 | |||||||
1170 | # warn "$self start STACK: ".$obj; | ||||||
1171 | $self->_process_comm($obj); | ||||||
1172 | |||||||
1173 | # warn "$self end STACK: ".$obj; | ||||||
1174 | } | ||||||
1175 | |||||||
1176 | # warn ref($self)."END PROCESS STACK ".$comm->local_name; | ||||||
1177 | |||||||
1178 | } | ||||||
1179 | elsif ( $comm->{type} eq 'EV_START_ELEMENT' ) { | ||||||
1180 | my $current_obj = $comm->{data}; | ||||||
1181 | |||||||
1182 | # warn "$self: ev_START".$current_obj->local_name; | ||||||
1183 | #register new namespaces | ||||||
1184 | my $changes = $current_obj->ns->get_changes; | ||||||
1185 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
1186 | |||||||
1187 | for ( keys %$changes ) { | ||||||
1188 | $self->end_prefix_mapping( | ||||||
1189 | { | ||||||
1190 | Prefix => $_, | ||||||
1191 | NamespaceURI => $parent_map->{$_}, | ||||||
1192 | } | ||||||
1193 | ) if exists $parent_map->{$_}; | ||||||
1194 | |||||||
1195 | $self->start_prefix_mapping( | ||||||
1196 | { | ||||||
1197 | Prefix => $_, | ||||||
1198 | NamespaceURI => $changes->{$_}, | ||||||
1199 | } | ||||||
1200 | ); | ||||||
1201 | } | ||||||
1202 | |||||||
1203 | #save element in stack | ||||||
1204 | push @{ $self->_objects_stack() }, $current_obj; | ||||||
1205 | |||||||
1206 | #warn ref($self) . ": <" . $comm->{data}->local_name . ">"; | ||||||
1207 | |||||||
1208 | #skip deleted elements from xml stream | ||||||
1209 | unless ( $current_obj->is_delete_element ) { | ||||||
1210 | if ( UNIVERSAL::isa( $self->{Handler}, 'XML::ExtOn' ) ) { | ||||||
1211 | my $cloned = $current_obj->__clone; | ||||||
1212 | unless ( $self->{__make_self_events} ) { | ||||||
1213 | $self->{Handler}->start_element($cloned); | ||||||
1214 | } | ||||||
1215 | else { | ||||||
1216 | $self->{Handler}->__start_element($cloned); | ||||||
1217 | |||||||
1218 | } | ||||||
1219 | } | ||||||
1220 | else { | ||||||
1221 | my $res_data = $self->__exp_element_to_sax2($current_obj); | ||||||
1222 | $self->SUPER::start_element($res_data); | ||||||
1223 | } | ||||||
1224 | } | ||||||
1225 | } | ||||||
1226 | elsif ( $comm->{type} eq 'EV_END_ELEMENT' ) { | ||||||
1227 | my $current_obj = $comm->{data}; | ||||||
1228 | my $current_obj1 = pop @{ $self->_objects_stack() }; | ||||||
1229 | |||||||
1230 | #warn "END_E: ".$current_obj->local_name; | ||||||
1231 | # if skip | ||||||
1232 | #check current element for skip_content | ||||||
1233 | # if ( my $current_element = $self->current_element ) { | ||||||
1234 | # my $skip_content = $current_element->is_skip_content; | ||||||
1235 | # if ( $skip_content > 1 ) { | ||||||
1236 | # $current_element->is_skip_content( --$skip_content ); | ||||||
1237 | # return; | ||||||
1238 | # } | ||||||
1239 | # } | ||||||
1240 | |||||||
1241 | unless ( $current_obj->is_delete_element ) { | ||||||
1242 | unless ( $self->{__make_self_events} ) { | ||||||
1243 | |||||||
1244 | #convert to SAX2 | ||||||
1245 | my $data = $current_obj->to_sax2; | ||||||
1246 | delete $data->{Attributes}; | ||||||
1247 | $data->{NamespaceURI} = $current_obj->default_uri; | ||||||
1248 | $self->SUPER::end_element($data); | ||||||
1249 | } | ||||||
1250 | else { | ||||||
1251 | |||||||
1252 | #call with object | ||||||
1253 | $self->{Handler}->__end_element($current_obj1); | ||||||
1254 | } | ||||||
1255 | } | ||||||
1256 | |||||||
1257 | my $changes = $current_obj->ns->get_changes; | ||||||
1258 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
1259 | for ( keys %$changes ) { | ||||||
1260 | $self->end_prefix_mapping( | ||||||
1261 | { | ||||||
1262 | Prefix => $_, | ||||||
1263 | NamespaceURI => $changes->{$_}, | ||||||
1264 | } | ||||||
1265 | ); | ||||||
1266 | if ( exists( $parent_map->{$_} ) ) { | ||||||
1267 | $self->start_prefix_mapping( | ||||||
1268 | { | ||||||
1269 | Prefix => $_, | ||||||
1270 | NamespaceURI => $parent_map->{$_}, | ||||||
1271 | } | ||||||
1272 | ); | ||||||
1273 | } | ||||||
1274 | } | ||||||
1275 | |||||||
1276 | #warn ref($self) . ": " . $comm->{data}->local_name . ">"; | ||||||
1277 | |||||||
1278 | } | ||||||
1279 | } | ||||||
1280 | else { | ||||||
1281 | warn " Unknown DATA $comm"; | ||||||
1282 | } | ||||||
1283 | } | ||||||
1284 | |||||||
1285 | =head2 add_namespace |
||||||
1286 | |||||||
1287 | Add Namespace mapping. return C<$self> | ||||||
1288 | |||||||
1289 | If C |
||||||
1290 | that have no prefix. | ||||||
1291 | |||||||
1292 | $elem->add_namespace( | ||||||
1293 | "myns" => 'http://example.com/myns', | ||||||
1294 | "myns_test", 'http://example.com/myns_test', | ||||||
1295 | ''=>'http://example.com/new_default_namespace' | ||||||
1296 | ); | ||||||
1297 | |||||||
1298 | =cut | ||||||
1299 | |||||||
1300 | sub add_namespace { | ||||||
1301 | my $self = shift; | ||||||
1302 | my $context = $self->context; | ||||||
1303 | if ( my $current = $self->current_element ) { | ||||||
1304 | $context = $current->ns; | ||||||
1305 | } | ||||||
1306 | my %map = @_; | ||||||
1307 | while ( my ( $prefix, $ns_uri ) = each %map ) { | ||||||
1308 | $context->declare_prefix( $prefix, $ns_uri ); | ||||||
1309 | } | ||||||
1310 | } | ||||||
1311 | |||||||
1312 | #overload sub parse | ||||||
1313 | |||||||
1314 | =head2 parse |
||||||
1315 | |||||||
1316 | |||||||
1317 | =cut | ||||||
1318 | |||||||
1319 | sub parse { | ||||||
1320 | my ( $self, $in ) = @_; | ||||||
1321 | my $sax2_filter = XML::Filter::SAX1toSAX2->new( Handler => $self ); | ||||||
1322 | my $parser = XML::Parser::PerlSAX->new( { Handler => $sax2_filter } ); | ||||||
1323 | unless ( ref($in) ) { | ||||||
1324 | |||||||
1325 | # $self->_process_comm( $self->mk_from_xml($in) ); | ||||||
1326 | $parser->parse( Source => { String => $in } ); | ||||||
1327 | } | ||||||
1328 | elsif (UNIVERSAL::isa( $in, 'IO::Handle' ) | ||||||
1329 | or ( ( ref $in ) eq 'GLOB' ) | ||||||
1330 | or UNIVERSAL::isa( $in, 'Tie::Handle' ) ) | ||||||
1331 | { | ||||||
1332 | $parser->parse( Source => { ByteStream => $in } ) | ||||||
1333 | |||||||
1334 | } | ||||||
1335 | else { | ||||||
1336 | die "unknown params"; | ||||||
1337 | } | ||||||
1338 | } | ||||||
1339 | |||||||
1340 | 1; | ||||||
1341 | __END__ |