XML-Parsepp-Testgen
view release on metacpan or search on metacpan
lib/XML/Parsepp/Testgen.pm view on Meta::CPAN
require Exporter;
our @ISA = qw(Exporter);
our @EXPORT = qw();
our @EXPORT_OK = qw(xml_2_test test_2_xml);
my $template;
sub xml_2_test {
my ($input, $opts) = @_;
my $xml = parm_2_text($input);
my ($xml_def1, $xml_def2) = $xml =~ m{\A (\N*) \n (\N*) \n}xms
or die "Error-0010: Can't extract xml_defs from '".
(substr($xml, 0, 50) =~ s{\n}'\\n'xmsgr)."...'";
unless ($xml_def1 eq '#! Testdata for XML::Parsepp') {
die "Error-0020: Expected xml_def1 to be '#! Testdata for XML::Parsepp', but found '$xml_def1'";
}
unless ($xml_def2 eq '#! Ver 0.01') {
die "Error-0030: Expected xml_def2 to be '#! Ver 0.01', but found '$xml_def2'";
}
# $check_positions = 1 ==> check error-positions of the following form:
# if ($err =~ m{at \s+ line \s+ (\d+), \s+ column \s+ (\d+), \s+ byte \s+ (\d+) \s+ at \s+}xms)
my $check_positions = defined($opts) ? $opts->{'chkpos'} : 0;
my @HList = (
[ 'Init', 'INIT', '(Expat)' ],
[ 'Final', 'FINL', '(Expat)' ],
[ 'Start', 'STRT', '(Expat, Element, @Attr)' ],
[ 'End', 'ENDL', '(Expat, Element)' ],
[ 'Char', 'CHAR', '(Expat, String)' ],
[ 'Proc', 'PROC', '(Expat, Target, Data)' ],
[ 'Comment', 'COMT', '(Expat, Data)' ],
[ 'CdataStart', 'CDST', '(Expat)' ],
[ 'CdataEnd', 'CDEN', '(Expat)' ],
[ 'Default', 'DEFT', '(Expat, String)' ],
[ 'Unparsed', 'UNPS', '(Expat, Entity, Base, Sysid, Pubid, Notation)' ],
[ 'Notation', 'NOTA', '(Expat, Notation, Base, Sysid, Pubid)' ],
# [ 'ExternEnt', 'EXEN', '(Expat, Base, Sysid, Pubid)' ],
# [ 'ExternEntFin', 'EXEF', '(Expat)' ],
[ 'Entity', 'ENTT', '(Expat, Name, Val, Sysid, Pubid, Ndata, IsParam)' ],
[ 'Element', 'ELEM', '(Expat, Name, Model)' ],
[ 'Attlist', 'ATTL', '(Expat, Elname, Attname, Type, Default, Fixed)' ],
[ 'Doctype', 'DOCT', '(Expat, Name, Sysid, Pubid, Internal)' ],
[ 'DoctypeFin', 'DOCF', '(Expat)' ],
[ 'XMLDecl', 'DECL', '(Expat, Version, Encoding, Standalone)' ],
);
my %HSub;
my $replm = q!s{([\x00-\x1f\[\]])}{sprintf('&<%02x>', ord($1))}xmsge!;
my $i = 0;
for my $hl (@HList) { $i++;
my $func_body = '';
my @vlist = split m{,}xms, $hl->[2] =~ s{[\s\(\)]}''xmsgr;
for my $vl (@vlist) {
$vl = '$'.$vl unless $vl =~ m{\A \@}xms;
}
$func_body .= "{ # ".sprintf('%2d', $i).". ".sprintf('%-15s', $hl->[0])." ".$hl->[2]."\n";
$func_body .= " my (".join(', ', @vlist).") = \@_;\n\n";
my $has_array = 0;
my $j = 0;
for my $vl (@vlist) { $j++;
next if $j == 1;
if ($vl =~ m{\A \@}xms) {
$has_array = 1;
$func_body .= " for my \$a ($vl) {\n";
$func_body .= " \$a //= '*undef*'; \$a =~ $replm;\n";
$func_body .= " }\n";
}
else {
$func_body .= " ".sprintf('%-12s', $vl)." //= '*undef*'; ".sprintf('%-12s', $vl).' =~ '.$replm.";\n";
}
}
$func_body .= "\n";
$func_body .= qq! local \$" = "], [";\n! if $has_array;
$func_body .= qq! push \@result, "!.$hl->[1];
$j = 0;
for my $vl (@vlist) { $j++;
next if $j == 1;
$func_body .= qq!,! unless $j == 2;
$func_body .= ' '.substr($vl, 1, 3)."=[$vl]";
}
$func_body .= qq!";\n!;
$func_body .= qq!}\n!;
$HSub{$hl->[0]} = $func_body;
}
my @result;
my $err = '';
my @HParam;
for my $hl (@HList) {
my $handler = eval 'sub '.$HSub{$hl->[0]};
if ($@) {
die "Error-0040: Can't eval 'sub ".$HSub{$hl->[0]}."' because $@";
}
unless (ref($handler) eq 'CODE') {
die "Error-0050: Expected ref(handler) = 'CODE', but found '".ref($handler)."'";
}
push @HParam, $hl->[0], $handler;
}
my $XmlParser = XML::Parser->new or die "Error-0060: Can't create XML::Parser -> new";
$XmlParser->setHandlers(@HParam);
my @current;
my @RList;
for (split m{\n}xms, $xml) {
if (m{\A \s* \#! (.*) \z}xms) {
my $remark = $1;
if ($remark =~ m{\A \s* =+ \s* \z}xms) {
push @RList, { xml => [@current] } if @current;
@current = ();
}
}
else {
s{\s+ \z}''xms;
push @current, $_;
}
}
push @RList, { xml => [@current] } if @current;
my %HitCount = map { $_->[1] => 0 } @HList;
my $TestCount = @HList + 1;
for my $rl (@RList) {
# get_result($XmlParser, map {"$_\n"} @{$rl->{xml}});
@result = ();
$err = '';
my $ExpatNB = $XmlParser->parse_start or die "Error-0070: Can't create XML::Parser -> parse_start";
eval {
for my $buf (map {"$_\n"} @{$rl->{xml}}) {
$ExpatNB->parse_more($buf);
}
};
if ($@) {
$err = $@;
$ExpatNB->release;
}
else {
eval {
$ExpatNB->parse_done;
};
if ($@) {
$err = $@;
}
}
$rl->{err} = $err;
$rl->{res} = [@result];
$TestCount += 2 + @result;
unless ($err eq '') {
$err =~ m{at \s+ line \s+ (\d+), \s+ column \s+ (\d+), \s+ byte \s+ (\d+) \s+ at \s+}xms
or die "Error-0080: Can't decompose error-line '$err'";
$rl->{e_line} = $1;
$rl->{e_col} = $2;
$rl->{e_bytes} = $3;
if ($check_positions) {
$TestCount += 3;
}
}
for my $res (@result) {
my $word = !defined($res) ? '!!!!' : $res =~ m{\A (\w{4}) }xms ? $1 : '????';
$HitCount{$word}++;
}
}
my $result = '';
open my $ofh, '>', \$result or die "Error-0090: Can't open > '\\\$result' because $!";
for (split m{\n}xms, $template) {
if (m{\A \s* %}xms) {
m{\A %include \s+ (\w+) \z}xms
or die "Error-0100: Can't parse %include from '$_'";
my $subject = $1;
if ($subject eq 'test_more') {
say {$ofh} "use Test::More tests => $TestCount;"
}
elsif ($subject eq 'handlers') {
my $ctr = 0;
for my $hl (@HList) { $ctr++;
printf {$ofh} " [%3d, %-12s => \\&handle_%-13s %-6s, occurs => %4d, %-65s ],\n",
$ctr, $hl->[0], $hl->[0].',', "'$hl->[1]'", $HitCount{$hl->[1]}, sprintf("'%-12s %s'", $hl->[0], $hl->[2]);
}
}
elsif ($subject eq 'cases') {
say {$ofh} '# No of get_result is ', scalar(@RList);
say {$ofh} '';
my $tno = 0;
for my $rl (@RList) { $tno++;
say {$ofh} '{';
say {$ofh} ' get_result($XmlParser,';
for my $lx (@{$rl->{xml}}) {
if ($lx =~ m![\\{}]!xms) {
die "Error-0110: Found invalid character in xml line '$lx'";
}
say {$ofh} " q{$lx}.qq{\\n},";
}
say {$ofh} ' );';
say {$ofh} '';
say {$ofh} ' my @expected = (';
for my $ls (@{$rl->{res}}) {
if ($ls =~ m![\\{}]!xms) {
die "Error-0120: Found invalid character in result line '$ls'";
}
say {$ofh} " q{$ls},";
}
say {$ofh} ' );';
say {$ofh} '';
my $ecode = $rl->{err};
if ($ecode eq '') {
say {$ofh} q{ is($err, '', 'Test-}, sprintf('%03d', $tno), q{a: No error');};
}
else {
$ecode =~ m{\A (.*?) \s+ at \s+ line \s+ \d+}xms
or die "Error-0130: Can't parse message from ecode = '$ecode'";
my $emsg = $1;
$emsg =~ s{\A \s+}''xms;
$emsg =~ m{\A [\w\s()\-]* \z}xms
lib/XML/Parsepp/Testgen.pm view on Meta::CPAN
}
}
else {
die "Error-0150: Found invalid %include subject '$subject'";
}
}
else {
say {$ofh} $_;
}
}
close $ofh;
return $result;
}
sub test_2_xml {
my ($input) = @_;
my $perl = parm_2_text($input);
my ($def1, $def2, $def3) = $perl =~ m{\A (\N*) \n (\N*) \n (\N*) \n}xms
or die "Error-0160: Can't extract use-statements from '".
(substr($perl, 0, 50) =~ s{\n}'\\n'xmsgr)."...'";
unless ($def1 eq 'use 5.014;') {
die "Error-0170: Expected def1 to be 'use 5.014;', but found '$def1'";
}
unless ($def2 eq 'use warnings;') {
die "Error-0180: Expected def2 to be 'use warnings;', but found '$def2'";
}
unless ($def3 eq '# Generate Tests for XML::Parsepp') {
die "Error-0190: Expected def3 to be '# Generate Tests for XML::Parsepp', but found '$def3'";
}
$perl =~ m{\n\# \s No \s of \s get_result \s is \s (\d+) \n}xms
or die "Error-0200: Can't find 'No of get_result...'";
my $gr_count = $1;
my @gr_list = $perl =~ m{get_result\( (.*?) \);}xmsg;
unless (@gr_list == $gr_count) {
die "Error-0210: Found ".scalar(@gr_list)." get_result, but expected $gr_count";
}
my $result = '';
open my $ofh, '>', \$result or die "Error-0220: Can't open > '\\\$result' because $!";
say {$ofh} '#! Testdata for XML::Parsepp';
say {$ofh} '#! Ver 0.01';
for my $i (0..$#gr_list) {
my $text = $gr_list[$i];
say {$ofh} '#! ===' unless $i == 0;
my @lines = split m{\n}xms, $text;
my $first = shift @lines;
unless (defined $first) {
die "Error-0230: Too few elements in lines";
}
unless ($first eq '$XmlParser,') {
die "Error-0240: found first line = >>$first<<, but expected >>\$XmlParser,<<";
}
for my $fragment (@lines) {
next if $fragment =~ m{\A \s* \z}xms;
my ($gr_xml) = $fragment =~ m/\A \s* q\{ ([^\}]*) \}\.qq\{\\n\}, \s* \z/xms
or do {
local $" = "<<, >>";
die "Error-0250: Can't parse fragment q{...} >>$fragment<<, all lines are (>>@lines<<)";
};
say {$ofh} $gr_xml;
}
}
close $ofh;
return $result;
}
sub parm_2_text {
my ($inp) = @_;
my $data;
if (ref($inp)) {
if (ref($inp) eq 'GLOB') {
$data = do { local $/; <$inp>; };
}
else {
$data = $$inp;
}
}
else {
open my $fh, '<', $inp or die "Error-0260: Can't open < '$inp' because $!";
$data = do { local $/; <$fh>; };
}
return $data;
}
$template = <<'EOTXT';
use 5.014;
use warnings;
# Generate Tests for XML::Parsepp
%include test_more
my $XML_module = 'XML::Parsepp';
use_ok($XML_module);
my @result;
( run in 2.535 seconds using v1.01-cache-2.11-cpan-71847e10f99 )