1 |
#!/usr/bin/perl -w |
2 |
use lib q<../lib>; |
3 |
use strict; |
4 |
BEGIN { require 'manakai/genlib.pl' } |
5 |
|
6 |
use Message::Util::QName::General [qw/ExpandedURI/], { |
7 |
ManakaiDOMLS2003 |
8 |
=> q<http://suika.fam.cx/~wakaba/archive/2004/9/27/mdom-old-ls#>, |
9 |
}; |
10 |
use Message::DOM::ManakaiDOMLS2003; |
11 |
use Message::DOM::DOMLS qw/MODE_SYNCHRONOUS/; |
12 |
use Getopt::Long; |
13 |
|
14 |
require 'dommemlist.pl.tmp'; ## Generated by mkdommemlist.pl |
15 |
|
16 |
my $output_filename; |
17 |
my $output_file; |
18 |
GetOptions ( |
19 |
'output-file=s' => \$output_filename, |
20 |
); |
21 |
if (defined $output_filename) { |
22 |
open $output_file, '>', $output_filename or die "$0: $output_filename: $!"; |
23 |
} else { |
24 |
$output_file = \*STDOUT; |
25 |
} |
26 |
|
27 |
our $Method; |
28 |
our $IFMethod; |
29 |
our $Attr; |
30 |
my $Assert = { |
31 |
qw/assertDOMException 1 |
32 |
assertFalse 1 |
33 |
assertNotNull 1 |
34 |
assertNull 1 |
35 |
assertSize 1 |
36 |
assertTrue 1/ |
37 |
}; |
38 |
my $Misc = { |
39 |
qw/if 1 |
40 |
implementationAttribute 1 |
41 |
var 1/ |
42 |
}; |
43 |
my $Condition = { |
44 |
qw/condition 1 |
45 |
contains 1 |
46 |
contentType 1 |
47 |
hasSize 1 |
48 |
implementationAttribute 1 |
49 |
not 1 |
50 |
notNull 1 |
51 |
or 1/ |
52 |
}; |
53 |
|
54 |
my $Status = {Number => 0, our => {Info => 1}}; |
55 |
|
56 |
## Defined in genlib.pl but redefined. |
57 |
sub output_result ($) { |
58 |
print $output_file shift; |
59 |
} |
60 |
|
61 |
sub to_perl_value ($;%) { |
62 |
my ($s, %opt) = @_; |
63 |
if (defined $s) { |
64 |
if ($s =~ /^(?!\d)\w+$/) { |
65 |
return perl_var (type => '$', local_name => $s); |
66 |
} else { |
67 |
return $s; |
68 |
} |
69 |
} elsif (defined $opt{default}) { |
70 |
return $opt{default}; |
71 |
} else { |
72 |
return ''; |
73 |
} |
74 |
} |
75 |
|
76 |
sub body2code ($) { |
77 |
my $parent = shift; |
78 |
my $result = ''; |
79 |
my $children = $parent->childNodes; |
80 |
for (my $i = 0; $i < $children->length; $i++) { |
81 |
my $child = $children->item ($i); |
82 |
if ($child->nodeType == $child->ELEMENT_NODE) { |
83 |
my $ln = $child->localName; |
84 |
if ($Method->{$ln} or $Attr->{$ln} or |
85 |
$Assert->{$ln} or $Misc->{$ln}) { |
86 |
$result .= node2code ($child); |
87 |
} else { |
88 |
valid_err q<Unknown element type: >.$child->localName, |
89 |
node => $child; |
90 |
} |
91 |
} elsif ($child->nodeType == $child->COMMENT_NODE) { |
92 |
$result .= perl_comment $child->data; |
93 |
} elsif ($child->nodeType == $child->TEXT_NODE) { |
94 |
if ($child->data =~ /\S/) { |
95 |
valid_err q<Unknown character data: >.$child->data, |
96 |
node => $child; |
97 |
} |
98 |
} else { |
99 |
valid_err q<Unknown type of node: >.$child->nodeType, |
100 |
node => $child; |
101 |
} |
102 |
} |
103 |
$result; |
104 |
} |
105 |
|
106 |
sub condition2code ($;%) { |
107 |
my ($parent, %opt) = @_; |
108 |
my $result = ''; |
109 |
my @result; |
110 |
my $children = $parent->childNodes; |
111 |
for (my $i = 0; $i < $children->length; $i++) { |
112 |
my $child = $children->item ($i); |
113 |
if ($child->nodeType == $child->ELEMENT_NODE) { |
114 |
my $ln = $child->localName; |
115 |
if ($Condition->{$ln}) { |
116 |
push @result, node2code ($child); |
117 |
} else { |
118 |
valid_err q<Unknown element type: >.$child->localName, |
119 |
node => $child; |
120 |
} |
121 |
} elsif ($child->nodeType == $child->COMMENT_NODE) { |
122 |
$result .= perl_comment $child->data; |
123 |
} elsif ($child->nodeType == $child->TEXT_NODE) { |
124 |
if ($child->data =~ /\S/) { |
125 |
valid_err q<Unknown character data: >.$child->data, |
126 |
node => $child; |
127 |
} |
128 |
} else { |
129 |
valid_err q<Unknown type of node: >.$child->nodeType, |
130 |
node => $child; |
131 |
} |
132 |
} |
133 |
$result .= join (($opt{join}||='or' eq 'or' ? ' || ' : |
134 |
$opt{join} eq 'and' ? ' && ' : |
135 |
valid_err q<Multiple condition not supported>, |
136 |
node => $parent), |
137 |
map {"($_)"} @result); |
138 |
$result; |
139 |
} #condition2code |
140 |
|
141 |
sub node2code ($); |
142 |
sub node2code ($) { |
143 |
my $node = shift; |
144 |
my $result = ''; |
145 |
if ($node->nodeType != $node->ELEMENT_NODE) { |
146 |
if ($node->nodeType == $node->COMMENT_NODE) { |
147 |
$result .= perl_comment $node->data; |
148 |
} elsif ($node->nodeType == $node->TEXT_NODE) { |
149 |
if ($node->data =~ /\S/) { |
150 |
valid_err q<Unknown character data: >.$node->data, |
151 |
node => $node; |
152 |
} |
153 |
} else { |
154 |
valid_err q<Unknown type of node: >.$node->nodeType, |
155 |
node => $node; |
156 |
} |
157 |
return $result; |
158 |
} |
159 |
my $ln = $node->localName; |
160 |
|
161 |
if ($ln eq 'var') { |
162 |
my $name = $node->getAttributeNS (undef, 'name'); |
163 |
$result .= perl_statement |
164 |
perl_var |
165 |
local_name => $name, |
166 |
scope => 'my', |
167 |
type => '$'; |
168 |
if ($node->getAttributeNS (undef, 'value')) { |
169 |
valid_err q<Attribute "value" not supported>, node => $node; |
170 |
} |
171 |
$Status->{var}->{$name}->{type} = $node->getAttributeNS (undef, 'type'); |
172 |
} elsif ($ln eq 'load') { |
173 |
$result .= perl_statement |
174 |
perl_assign |
175 |
perl_var |
176 |
(type => '$', |
177 |
local_name => $node->getAttributeNS (undef, 'var')) |
178 |
=> 'load (' . |
179 |
perl_literal ($node->getAttributeNS (undef, 'href')). |
180 |
')'; |
181 |
} elsif ($Method->{$ln}) { |
182 |
$result .= perl_var (type => '$', |
183 |
local_name => $node->getAttributeNS (undef, 'var')). |
184 |
' = ' |
185 |
if $node->hasAttributeNS (undef, 'var'); |
186 |
my $param; |
187 |
if ($node->hasAttributeNS (undef, 'interface')) { |
188 |
$param = $IFMethod->{$node->getAttributeNS (undef, 'interface')} |
189 |
->{$ln}; |
190 |
} else { |
191 |
$param = $Method->{$ln}; |
192 |
} |
193 |
$result .= perl_var (type => '$', |
194 |
local_name => $node->getAttributeNS (undef, 'obj')). |
195 |
'->'.$ln.' ('. |
196 |
join (', ', |
197 |
map { |
198 |
to_perl_value ($node->getAttributeNS (undef, $_), |
199 |
default => 'undef') |
200 |
} @$param). |
201 |
");\n"; |
202 |
} elsif ($Attr->{$ln}) { |
203 |
if ($node->hasAttributeNS (undef, 'var')) { |
204 |
$result .= perl_var (type => '$', |
205 |
local_name => $node->getAttributeNS (undef, 'var')). |
206 |
' = '; |
207 |
} elsif ($node->hasAttributeNS (undef, 'value')) { |
208 |
# |
209 |
} else { |
210 |
valid_err q<Unknown operation to an attribute>, node => $node; |
211 |
} |
212 |
$result .= perl_var (type => '$', |
213 |
local_name => $node->getAttributeNS (undef, 'obj')). |
214 |
'->'.$ln; |
215 |
if ($node->hasAttributeNS (undef, 'var')) { |
216 |
$result .= ";\n"; |
217 |
} elsif ($node->hasAttributeNS (undef, 'value')) { |
218 |
$result .= " (".to_perl_value ($node->getAttributeNS (undef, 'value')). |
219 |
");\n"; |
220 |
} |
221 |
} elsif ($ln eq 'assertEquals') { |
222 |
my $expected = $node->getAttributeNS (undef, 'expected'); |
223 |
my $expectedType = $Status->{var}->{$expected}->{type} || ''; |
224 |
$result .= 'assertEquals'. |
225 |
({Collection => 'Collection', |
226 |
List => 'List'}->{$expectedType}||''); |
227 |
my $ignoreCase = $node->getAttributeNS (undef, 'ignoreCase') || 'false'; |
228 |
if ($ignoreCase eq 'auto') { |
229 |
$result .= 'AutoCase ('. |
230 |
perl_literal ($node->getAttributeNS (undef, 'context') || |
231 |
'element'). |
232 |
', '; |
233 |
} else { |
234 |
$result .= ' ('; |
235 |
} |
236 |
$result .= perl_literal ($node->getAttributeNS (undef, 'id')).', '; |
237 |
$result .= join ", ", map { |
238 |
$ignoreCase eq 'true' |
239 |
? ($expectedType eq 'Collection' or |
240 |
$expectedType eq 'List') |
241 |
? "toLowerArray ($_)" : "lc ($_)" |
242 |
: $_ |
243 |
} map { |
244 |
to_perl_value ($_) |
245 |
} ( |
246 |
$expected, |
247 |
$node->getAttributeNS (undef, 'actual'), |
248 |
); |
249 |
$result .= ");\n"; |
250 |
$Status->{Number}++; |
251 |
} elsif ($ln eq 'assertSize') { |
252 |
my $size = to_perl_value ($node->getAttributeNS (undef, 'size')); |
253 |
my $coll = to_perl_value ($node->getAttributeNS (undef, 'collection')); |
254 |
$result .= perl_statement 'assertSize ('. |
255 |
perl_list |
256 |
($node->getAttributeNS (undef, 'id'), |
257 |
perl_code_literal $size, perl_code_literal $coll). |
258 |
')'; |
259 |
if ($node->hasChildNodes) { |
260 |
$result .= perl_if |
261 |
qq<$size == size ($coll)>, |
262 |
block2code ($node); |
263 |
} |
264 |
} elsif ($ln eq 'assertTrue' or $ln eq 'assertFalse') { |
265 |
my $condition; |
266 |
if ($node->hasAttributeNS (undef, 'actual')) { |
267 |
$condition = perl_var (type => '$', |
268 |
local_name => $node->getAttributeNS |
269 |
(undef, 'actual')); |
270 |
if ($node->hasChildNodes) { |
271 |
valid_err q<Child of $ln found but not supported>, |
272 |
node => $node; |
273 |
} |
274 |
} elsif ($node->hasChildNodes) { |
275 |
$condition = condition2code ($node); |
276 |
} else { |
277 |
valid_err $ln.q< w/o @actual not supported>, node => $node; |
278 |
} |
279 |
$result .= perl_statement $ln . ' ('. |
280 |
perl_literal ($node->getAttributeNS (undef, 'id')).', '. |
281 |
$condition. ')'; |
282 |
$Status->{Number}++; |
283 |
} elsif ($ln eq 'assertNotNull' or $ln eq 'assertNull') { |
284 |
$result .= perl_statement $ln . ' (' . |
285 |
perl_literal ($node->getAttributeNS (undef, 'id')).', '. |
286 |
perl_var (type => '$', |
287 |
local_name => $node->getAttributeNS (undef, 'actual')). |
288 |
')'; |
289 |
if ($node->hasChildNodes) { |
290 |
valid_err q<Child of $ln found but not supported>, |
291 |
node => $node; |
292 |
} |
293 |
$Status->{Number}++; |
294 |
} elsif ($ln eq 'assertDOMException') { |
295 |
$Status->{use}->{'Message::Util::Error'} = 1; |
296 |
$result .= q[ |
297 |
{ |
298 |
my $success = 0; |
299 |
try { |
300 |
]; |
301 |
my $children = $node->childNodes; |
302 |
my $errname; |
303 |
for (my $i = 0; $i < $children->length; $i++) { |
304 |
my $child = $children->item ($i); |
305 |
$errname = $child->localName if $child->nodeType == $child->ELEMENT_NODE; |
306 |
$result .= body2code ($child); |
307 |
} |
308 |
$result .= q[ |
309 |
} catch Message::DOM::DOMException with { |
310 |
my $err = shift; |
311 |
$success = 1 if $err->{-type} eq ].perl_literal ($errname).q[; |
312 |
} |
313 |
assertTrue (].perl_literal ($node->getAttributeNS (undef, 'id')). |
314 |
q[, $success); |
315 |
} |
316 |
]; |
317 |
$Status->{Number}++; |
318 |
} elsif ($ln eq 'contentType') { |
319 |
$result .= '$builder->{contentType} eq '. |
320 |
perl_literal ($node->getAttributeNS (undef, 'type')); |
321 |
$Status->{our}->{builder} = 1; |
322 |
} elsif ($ln eq 'if') { |
323 |
my $children = $node->childNodes; |
324 |
my $condition; |
325 |
my $true = ''; |
326 |
for (my $i = 0; $i < $children->length; $i++) { |
327 |
my $child = $children->item ($i); |
328 |
if ($child->nodeType == $child->ELEMENT_NODE) { |
329 |
if (not $condition) { |
330 |
$condition = node2code ($child); |
331 |
} elsif ($child->localName eq 'else') { |
332 |
valid_err q<Multiple 'else's found>, node => $child |
333 |
if $true; |
334 |
$true = $result; |
335 |
$result = ''; |
336 |
} else { |
337 |
$result .= node2code ($child); |
338 |
} |
339 |
} else { |
340 |
$result .= node2code ($child); |
341 |
} |
342 |
} |
343 |
$result = perl_if |
344 |
$condition, |
345 |
$true || $result, |
346 |
$true ? $result : undef; |
347 |
} elsif ($ln eq 'or') { |
348 |
$result .= condition2code ($node, join => 'or'); |
349 |
} elsif ($ln eq 'not') { |
350 |
$result .= 'not '.condition2code ($node, join => 'nosupport'); |
351 |
} elsif ($ln eq 'notNull') { |
352 |
$result .= 'defined '. |
353 |
perl_var (type => '$', |
354 |
local_name => $node->getAttributeNS (undef, 'obj')); |
355 |
} else { |
356 |
valid_err q<Unknown element type: >.$ln; |
357 |
} |
358 |
$result; |
359 |
} |
360 |
|
361 |
our $result = ''; |
362 |
|
363 |
my $input; |
364 |
{ |
365 |
local $/ = undef; |
366 |
$input = <>; |
367 |
} |
368 |
|
369 |
{ |
370 |
my $dom = Message::DOM::DOMImplementationRegistry |
371 |
->getDOMImplementation |
372 |
({Core => undef, |
373 |
XML => undef, |
374 |
ExpandedURI q<ManakaiDOMLS2003:LS> => '1.0'}); |
375 |
|
376 |
my $parser = $dom->createLSParser (MODE_SYNCHRONOUS); |
377 |
my $in = $dom->createLSInput; |
378 |
$in->stringData ($input); |
379 |
|
380 |
my $src = $parser->parse ($in)->documentElement; |
381 |
|
382 |
{ |
383 |
my $children = $src->ownerDocument->childNodes; |
384 |
for (my $i = 0; $i < $children->length; $i++) { |
385 |
my $node = $children->item ($i); |
386 |
if ($node->nodeType == $node->COMMENT_NODE) { |
387 |
if ($node->data =~ /Copyright/) { |
388 |
$result .= perl_comment |
389 |
qq<This script was generated by "$0"\n>. |
390 |
qq<and is a derived work from the source document.\n>. |
391 |
qq<The source document contained the following notice:\n>. |
392 |
$node->data; |
393 |
} else { |
394 |
$result .= perl_comment $node->data; |
395 |
} |
396 |
} |
397 |
} |
398 |
} |
399 |
|
400 |
my $child = $src->childNodes; |
401 |
|
402 |
for (my $i = 0; $i < $child->length; $i++) { |
403 |
my $node = $child->item ($i); |
404 |
if ($node->nodeType == $node->ELEMENT_NODE) { |
405 |
my $ln = $node->localName; |
406 |
if ($ln eq 'metadata') { |
407 |
my $md = $node->childNodes; |
408 |
for (my $j = 0; $j < $md->length; $j++) { |
409 |
my $node = $md->item ($j); |
410 |
if ($node->nodeType == $node->ELEMENT_NODE) { |
411 |
my $ln = $node->localName; |
412 |
if ($ln eq 'title') { |
413 |
$result .= perl_statement |
414 |
perl_assign |
415 |
'$Info->{Name}' |
416 |
=> perl_literal $node->textContent; |
417 |
} elsif ($ln eq 'description') { |
418 |
$result .= perl_statement |
419 |
perl_assign |
420 |
'$Info->{Description}' |
421 |
=> perl_literal $node->textContent; |
422 |
} else { |
423 |
# valid_err q<Unknown element type: >.$ln, |
424 |
# node => $node; |
425 |
} |
426 |
} elsif ($node->nodeType == $node->TEXT_NODE) { |
427 |
if ($node->data =~ /\S/) { |
428 |
valid_err q<Unknown character data: >.$node->data, |
429 |
node => $node; |
430 |
} |
431 |
} elsif ($node->nodeType == $node->COMMENT_NODE) { |
432 |
$result .= perl_comment $node->data; |
433 |
} else { |
434 |
valid_err q<Unknown node type: >.$node->nodeType, |
435 |
node => $node; |
436 |
} |
437 |
} |
438 |
} elsif ($ln eq 'implementationAttribute') { |
439 |
$result .= perl_comment |
440 |
sprintf 'Implementation attribute: @name=%s, @value=%s', |
441 |
$node->getAttributeNS (undef, 'name'), |
442 |
$node->getAttributeNS (undef, 'value'); |
443 |
} else { |
444 |
$result .= node2code ($node); |
445 |
} |
446 |
} elsif ($node->nodeType == $node->COMMENT_NODE) { |
447 |
$result .= perl_comment $node->data; |
448 |
} elsif ($node->nodeType == $node->TEXT_NODE) { |
449 |
if ($node->data =~ /\S/) { |
450 |
valid_err q<Unknown character data: >.$node->data, |
451 |
node => $node; |
452 |
} |
453 |
} else { |
454 |
valid_err q<Unknown type of node: >.$node->nodeType, |
455 |
node => $node; |
456 |
} |
457 |
} |
458 |
} |
459 |
|
460 |
my $pre = "#!/usr/bin/perl -w\nuse strict;\n"; |
461 |
$pre .= perl_statement ('require '.perl_literal 'manakai/domtest.pl'); |
462 |
$pre .= perl_statement |
463 |
('use Message::Util::Error') |
464 |
if $Status->{use}->{'Message::Util::Error'}; |
465 |
for (keys %{$Status->{our}}) { |
466 |
$pre .= perl_statement perl_var type => '$', local_name => $_, |
467 |
scope => 'our'; |
468 |
} |
469 |
$pre .= perl_statement q<plan (>.(0+$Status->{Number}).q<)>; |
470 |
|
471 |
output_result $pre.$result; |
472 |
|