2011-03-23 05:48:11 +08:00
|
|
|
#***************************************************************************
|
|
|
|
# _ _ ____ _
|
|
|
|
# Project ___| | | | _ \| |
|
|
|
|
# / __| | | | |_) | |
|
|
|
|
# | (__| |_| | _ <| |___
|
|
|
|
# \___|\___/|_| \_\_____|
|
|
|
|
#
|
2023-01-02 20:51:48 +08:00
|
|
|
# Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
|
2011-03-23 05:48:11 +08:00
|
|
|
#
|
|
|
|
# This software is licensed as described in the file COPYING, which
|
|
|
|
# you should have received as part of this distribution. The terms
|
2020-11-04 21:02:01 +08:00
|
|
|
# are also available at https://curl.se/docs/copyright.html.
|
2011-03-23 05:48:11 +08:00
|
|
|
#
|
|
|
|
# You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
|
|
# copies of the Software, and permit persons to whom the Software is
|
|
|
|
# furnished to do so, under the terms of the COPYING file.
|
|
|
|
#
|
|
|
|
# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
# KIND, either express or implied.
|
|
|
|
#
|
|
|
|
# SPDX-License-Identifier: curl
|
2022-05-17 17:16:50 +08:00
|
|
|
#
|
2011-03-23 05:48:11 +08:00
|
|
|
###########################################################################
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2023-04-06 03:28:26 +08:00
|
|
|
package getpart;
|
|
|
|
|
2023-03-28 10:58:00 +08:00
|
|
|
use strict;
|
|
|
|
use warnings;
|
2023-04-06 03:28:26 +08:00
|
|
|
|
|
|
|
BEGIN {
|
|
|
|
use base qw(Exporter);
|
|
|
|
|
|
|
|
our @EXPORT = qw(
|
|
|
|
getpartattr
|
|
|
|
getpart
|
|
|
|
partexists
|
|
|
|
loadtest
|
|
|
|
fulltest
|
|
|
|
striparray
|
|
|
|
compareparts
|
|
|
|
writearray
|
|
|
|
loadarray
|
|
|
|
showdiff
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-03-25 04:08:06 +08:00
|
|
|
use Memoize;
|
|
|
|
use MIME::Base64;
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2023-03-25 04:08:06 +08:00
|
|
|
my @xml; # test data file contents
|
|
|
|
my $xmlfile; # test data file name
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2002-01-08 17:32:10 +08:00
|
|
|
my $warning=0;
|
|
|
|
my $trace=0;
|
|
|
|
|
2023-03-25 04:08:06 +08:00
|
|
|
# Normalize the part function arguments for proper caching. This includes the
|
|
|
|
# file name in the arguments since that is an implied parameter that affects the
|
|
|
|
# return value. Any error messages will only be displayed the first time, but
|
|
|
|
# those are disabled by default anyway, so should never been seen outside
|
|
|
|
# development.
|
|
|
|
sub normalize_part {
|
|
|
|
push @_, $xmlfile;
|
|
|
|
return join("\t", @_);
|
|
|
|
}
|
2004-11-30 17:27:11 +08:00
|
|
|
|
2020-04-14 17:19:12 +08:00
|
|
|
sub decode_hex {
|
|
|
|
my $s = $_;
|
|
|
|
# remove everything not hex
|
|
|
|
$s =~ s/[^A-Fa-f0-9]//g;
|
|
|
|
# encode everything
|
|
|
|
$s =~ s/([a-fA-F0-9][a-fA-F0-9])/chr(hex($1))/eg;
|
|
|
|
return $s;
|
|
|
|
}
|
|
|
|
|
2022-11-29 23:41:15 +08:00
|
|
|
sub testcaseattr {
|
|
|
|
my %hash;
|
|
|
|
for(@xml) {
|
|
|
|
if(($_ =~ /^ *\<testcase ([^>]*)/)) {
|
|
|
|
my $attr=$1;
|
|
|
|
while($attr =~ s/ *([^=]*)= *(\"([^\"]*)\"|([^\> ]*))//) {
|
|
|
|
my ($var, $cont)=($1, $2);
|
|
|
|
$cont =~ s/^\"(.*)\"$/$1/;
|
|
|
|
$hash{$var}=$cont;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return %hash;
|
|
|
|
}
|
|
|
|
|
2001-05-29 05:49:45 +08:00
|
|
|
sub getpartattr {
|
2002-05-22 06:20:52 +08:00
|
|
|
# if $part is undefined (ie only one argument) then
|
|
|
|
# return the attributes of the section
|
|
|
|
|
2001-05-29 05:49:45 +08:00
|
|
|
my ($section, $part)=@_;
|
|
|
|
|
|
|
|
my %hash;
|
|
|
|
my $inside=0;
|
|
|
|
|
|
|
|
# print "Section: $section, part: $part\n";
|
|
|
|
|
|
|
|
for(@xml) {
|
|
|
|
# print "$inside: $_";
|
|
|
|
if(!$inside && ($_ =~ /^ *\<$section/)) {
|
|
|
|
$inside++;
|
|
|
|
}
|
2022-06-08 22:32:46 +08:00
|
|
|
if((1 ==$inside) && ( ($_ =~ /^ *\<$part ([^>]*)/) ||
|
2002-05-22 06:20:52 +08:00
|
|
|
!(defined($part)) )
|
|
|
|
) {
|
2001-05-29 05:49:45 +08:00
|
|
|
$inside++;
|
|
|
|
my $attr=$1;
|
|
|
|
|
2013-01-22 06:20:09 +08:00
|
|
|
while($attr =~ s/ *([^=]*)= *(\"([^\"]*)\"|([^\> ]*))//) {
|
2005-04-29 05:05:40 +08:00
|
|
|
my ($var, $cont)=($1, $2);
|
|
|
|
$cont =~ s/^\"(.*)\"$/$1/;
|
|
|
|
$hash{$var}=$cont;
|
2001-05-29 05:49:45 +08:00
|
|
|
}
|
|
|
|
last;
|
|
|
|
}
|
2011-08-04 23:36:31 +08:00
|
|
|
# detect end of section when part wasn't found
|
|
|
|
elsif((1 ==$inside) && ($_ =~ /^ *\<\/$section\>/)) {
|
|
|
|
last;
|
|
|
|
}
|
2001-05-29 05:49:45 +08:00
|
|
|
elsif((2 ==$inside) && ($_ =~ /^ *\<\/$part/)) {
|
|
|
|
$inside--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return %hash;
|
|
|
|
}
|
2023-03-25 04:08:06 +08:00
|
|
|
memoize('getpartattr', NORMALIZER => 'normalize_part'); # cache each result
|
2001-05-29 05:49:45 +08:00
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
sub getpart {
|
|
|
|
my ($section, $part)=@_;
|
|
|
|
|
|
|
|
my @this;
|
|
|
|
my $inside=0;
|
2004-11-29 20:10:09 +08:00
|
|
|
my $base64=0;
|
2020-04-14 17:19:12 +08:00
|
|
|
my $hex=0;
|
2020-03-11 00:47:44 +08:00
|
|
|
my $line;
|
2001-05-23 23:02:58 +08:00
|
|
|
|
|
|
|
for(@xml) {
|
2020-03-11 00:47:44 +08:00
|
|
|
$line++;
|
2001-05-23 23:02:58 +08:00
|
|
|
if(!$inside && ($_ =~ /^ *\<$section/)) {
|
|
|
|
$inside++;
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
elsif(($inside >= 1) && ($_ =~ /^ *\<$part[ \>]/)) {
|
|
|
|
if($inside > 1) {
|
|
|
|
push @this, $_;
|
|
|
|
}
|
|
|
|
elsif($_ =~ /$part [^>]*base64=/) {
|
|
|
|
# attempt to detect our base64 encoded part
|
2004-11-29 20:10:09 +08:00
|
|
|
$base64=1;
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
elsif($_ =~ /$part [^>]*hex=/) {
|
|
|
|
# attempt to detect a hex-encoded part
|
|
|
|
$hex=1;
|
|
|
|
}
|
2001-05-23 23:02:58 +08:00
|
|
|
$inside++;
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
elsif(($inside >= 2) && ($_ =~ /^ *\<\/$part[ \>]/)) {
|
|
|
|
if($inside > 2) {
|
|
|
|
push @this, $_;
|
|
|
|
}
|
2001-05-23 23:02:58 +08:00
|
|
|
$inside--;
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
elsif(($inside >= 1) && ($_ =~ /^ *\<\/$section/)) {
|
2020-03-11 00:47:44 +08:00
|
|
|
if($inside > 1) {
|
|
|
|
print STDERR "$xmlfile:$line:1: error: missing </$part> tag before </$section>\n";
|
|
|
|
@this = ("format error in $xmlfile");
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
if($trace && @this) {
|
2002-01-08 17:32:10 +08:00
|
|
|
print STDERR "*** getpart.pm: $section/$part returned data!\n";
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
if($warning && !@this) {
|
2002-01-08 17:32:10 +08:00
|
|
|
print STDERR "*** getpart.pm: $section/$part returned empty!\n";
|
|
|
|
}
|
2004-11-29 20:10:09 +08:00
|
|
|
if($base64) {
|
|
|
|
# decode the whole array before returning it!
|
|
|
|
for(@this) {
|
|
|
|
my $decoded = decode_base64($_);
|
|
|
|
$_ = $decoded;
|
|
|
|
}
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
elsif($hex) {
|
|
|
|
# decode the whole array before returning it!
|
|
|
|
for(@this) {
|
|
|
|
my $decoded = decode_hex($_);
|
|
|
|
$_ = $decoded;
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 23:02:58 +08:00
|
|
|
return @this;
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
elsif($inside >= 2) {
|
2001-05-23 23:02:58 +08:00
|
|
|
push @this, $_;
|
|
|
|
}
|
|
|
|
}
|
2012-06-21 02:13:07 +08:00
|
|
|
if($trace && @this) {
|
|
|
|
# section/part has data but end of section not detected,
|
|
|
|
# end of file implies end of section.
|
|
|
|
print STDERR "*** getpart.pm: $section/$part returned data!\n";
|
|
|
|
}
|
|
|
|
if($warning && !@this) {
|
|
|
|
# section/part does not exist or has no data without an end of
|
|
|
|
# section; end of file implies end of section.
|
2002-01-08 17:32:10 +08:00
|
|
|
print STDERR "*** getpart.pm: $section/$part returned empty!\n";
|
|
|
|
}
|
2012-06-22 00:17:50 +08:00
|
|
|
return @this;
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
2023-03-25 04:08:06 +08:00
|
|
|
memoize('getpart', NORMALIZER => 'normalize_part'); # cache each result
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2012-06-06 02:01:23 +08:00
|
|
|
sub partexists {
|
|
|
|
my ($section, $part)=@_;
|
|
|
|
|
|
|
|
my $inside = 0;
|
|
|
|
|
|
|
|
for(@xml) {
|
|
|
|
if(!$inside && ($_ =~ /^ *\<$section/)) {
|
|
|
|
$inside++;
|
|
|
|
}
|
|
|
|
elsif((1 == $inside) && ($_ =~ /^ *\<$part[ \>]/)) {
|
|
|
|
return 1; # exists
|
|
|
|
}
|
|
|
|
elsif((1 == $inside) && ($_ =~ /^ *\<\/$section/)) {
|
|
|
|
return 0; # does not exist
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0; # does not exist
|
|
|
|
}
|
2023-03-25 04:08:06 +08:00
|
|
|
# The code currently never calls this more than once per part per file, so
|
|
|
|
# caching a result that will never be used again just slows things down.
|
|
|
|
# memoize('partexists', NORMALIZER => 'normalize_part'); # cache each result
|
2012-06-06 02:01:23 +08:00
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
sub loadtest {
|
|
|
|
my ($file)=@_;
|
|
|
|
|
2023-04-21 08:41:31 +08:00
|
|
|
if(defined $xmlfile && $file eq $xmlfile) {
|
|
|
|
# This test is already loaded
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
undef @xml;
|
2023-03-25 04:08:06 +08:00
|
|
|
$xmlfile = "";
|
2004-02-26 17:19:16 +08:00
|
|
|
|
2023-03-29 04:29:36 +08:00
|
|
|
if(open(my $xmlh, "<", "$file")) {
|
|
|
|
binmode $xmlh; # for crapage systems, use binary
|
|
|
|
while(<$xmlh>) {
|
2004-02-26 17:19:16 +08:00
|
|
|
push @xml, $_;
|
|
|
|
}
|
2023-03-29 04:29:36 +08:00
|
|
|
close($xmlh);
|
2004-02-26 17:19:16 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
# failure
|
2004-02-27 06:56:32 +08:00
|
|
|
if($warning) {
|
|
|
|
print STDERR "file $file wouldn't open!\n";
|
|
|
|
}
|
2004-02-26 17:19:16 +08:00
|
|
|
return 1;
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
2023-03-25 04:08:06 +08:00
|
|
|
$xmlfile = $file;
|
2001-05-23 23:02:58 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-04-06 03:28:26 +08:00
|
|
|
|
|
|
|
# Return entire document as list of lines
|
2020-04-17 15:58:42 +08:00
|
|
|
sub fulltest {
|
|
|
|
return @xml;
|
|
|
|
}
|
|
|
|
|
|
|
|
# write the test to the given file
|
|
|
|
sub savetest {
|
|
|
|
my ($file)=@_;
|
|
|
|
|
2023-03-29 04:29:36 +08:00
|
|
|
if(open(my $xmlh, ">", "$file")) {
|
|
|
|
binmode $xmlh; # for crapage systems, use binary
|
2020-04-17 15:58:42 +08:00
|
|
|
for(@xml) {
|
2023-03-29 04:29:36 +08:00
|
|
|
print $xmlh $_;
|
2020-04-17 15:58:42 +08:00
|
|
|
}
|
2023-03-29 04:29:36 +08:00
|
|
|
close($xmlh);
|
2020-04-17 15:58:42 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
# failure
|
|
|
|
if($warning) {
|
|
|
|
print STDERR "file $file wouldn't open!\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
#
|
|
|
|
# Strip off all lines that match the specified pattern and return
|
|
|
|
# the new array.
|
|
|
|
#
|
|
|
|
|
|
|
|
sub striparray {
|
|
|
|
my ($pattern, $arrayref) = @_;
|
|
|
|
|
|
|
|
my @array;
|
|
|
|
|
|
|
|
for(@$arrayref) {
|
|
|
|
if($_ !~ /$pattern/) {
|
|
|
|
push @array, $_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return @array;
|
|
|
|
}
|
|
|
|
|
|
|
|
#
|
|
|
|
# pass array *REFERENCES* !
|
|
|
|
#
|
|
|
|
sub compareparts {
|
|
|
|
my ($firstref, $secondref)=@_;
|
|
|
|
|
2004-11-30 17:53:53 +08:00
|
|
|
my $first = join("", @$firstref);
|
|
|
|
my $second = join("", @$secondref);
|
2004-11-29 20:10:09 +08:00
|
|
|
|
|
|
|
# we cannot compare arrays index per index since with the base64 chunks,
|
|
|
|
# they may not be "evenly" distributed
|
|
|
|
|
2013-04-07 06:28:15 +08:00
|
|
|
# NOTE: this no longer strips off carriage returns from the arrays. Is that
|
|
|
|
# really necessary? It ruins the testing of newlines. I believe it was once
|
|
|
|
# added to enable tests on win32.
|
2004-11-29 20:10:09 +08:00
|
|
|
|
|
|
|
if($first ne $second) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#
|
|
|
|
# Write a given array to the specified file
|
|
|
|
#
|
|
|
|
sub writearray {
|
|
|
|
my ($filename, $arrayref)=@_;
|
|
|
|
|
2023-03-29 04:29:36 +08:00
|
|
|
open(my $temp, ">", "$filename") || die "Failure writing file";
|
|
|
|
binmode($temp,":raw"); # cygwin fix by Kevin Roth
|
2001-05-23 23:02:58 +08:00
|
|
|
for(@$arrayref) {
|
2023-03-29 04:29:36 +08:00
|
|
|
print $temp $_;
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
2023-03-29 04:29:36 +08:00
|
|
|
close($temp) || die "Failure writing file";
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#
|
2014-06-15 23:21:14 +08:00
|
|
|
# Load a specified file and return it as an array
|
2001-05-23 23:02:58 +08:00
|
|
|
#
|
|
|
|
sub loadarray {
|
|
|
|
my ($filename)=@_;
|
|
|
|
my @array;
|
|
|
|
|
2023-03-29 12:07:06 +08:00
|
|
|
if (open(my $temp, "<", "$filename")) {
|
|
|
|
while(<$temp>) {
|
|
|
|
push @array, $_;
|
|
|
|
}
|
|
|
|
close($temp);
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
|
|
|
return @array;
|
|
|
|
}
|
|
|
|
|
2004-11-30 17:53:53 +08:00
|
|
|
# Given two array references, this function will store them in two temporary
|
|
|
|
# files, run 'diff' on them, store the result and return the diff output!
|
|
|
|
|
2001-05-23 23:02:58 +08:00
|
|
|
sub showdiff {
|
2004-11-30 17:53:53 +08:00
|
|
|
my ($logdir, $firstref, $secondref)=@_;
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2004-11-30 17:53:53 +08:00
|
|
|
my $file1="$logdir/check-generated";
|
|
|
|
my $file2="$logdir/check-expected";
|
2010-02-15 03:40:18 +08:00
|
|
|
|
2023-03-29 04:29:36 +08:00
|
|
|
open(my $temp, ">", "$file1") || die "Failure writing diff file";
|
2001-05-23 23:02:58 +08:00
|
|
|
for(@$firstref) {
|
2014-01-26 17:00:14 +08:00
|
|
|
my $l = $_;
|
|
|
|
$l =~ s/\r/[CR]/g;
|
|
|
|
$l =~ s/\n/[LF]/g;
|
2021-05-19 15:37:01 +08:00
|
|
|
$l =~ s/([^\x20-\x7f])/sprintf "%%%02x", ord $1/eg;
|
2023-03-29 04:29:36 +08:00
|
|
|
print $temp $l;
|
|
|
|
print $temp "\n";
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
2023-03-29 04:29:36 +08:00
|
|
|
close($temp) || die "Failure writing diff file";
|
2001-05-23 23:02:58 +08:00
|
|
|
|
2023-03-29 04:29:36 +08:00
|
|
|
open($temp, ">", "$file2") || die "Failure writing diff file";
|
2001-05-23 23:02:58 +08:00
|
|
|
for(@$secondref) {
|
2014-01-26 17:00:14 +08:00
|
|
|
my $l = $_;
|
|
|
|
$l =~ s/\r/[CR]/g;
|
|
|
|
$l =~ s/\n/[LF]/g;
|
2021-05-19 15:37:01 +08:00
|
|
|
$l =~ s/([^\x20-\x7f])/sprintf "%%%02x", ord $1/eg;
|
2023-03-29 04:29:36 +08:00
|
|
|
print $temp $l;
|
|
|
|
print $temp "\n";
|
2001-05-23 23:02:58 +08:00
|
|
|
}
|
2023-03-29 04:29:36 +08:00
|
|
|
close($temp) || die "Failure writing diff file";
|
2005-04-28 16:20:33 +08:00
|
|
|
my @out = `diff -u $file2 $file1 2>/dev/null`;
|
|
|
|
|
|
|
|
if(!$out[0]) {
|
2010-02-16 21:32:45 +08:00
|
|
|
@out = `diff -c $file2 $file1 2>/dev/null`;
|
2005-04-28 16:20:33 +08:00
|
|
|
}
|
2001-05-23 23:02:58 +08:00
|
|
|
|
|
|
|
return @out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
1;
|