-
Notifications
You must be signed in to change notification settings - Fork 34
/
Copy pathseterrfile.pl
executable file
·116 lines (99 loc) · 3.16 KB
/
seterrfile.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#! /usr/local/bin/perl
# seterrfile
# C. DeTar 26 Mar 2005
# Constructs a tentative error tolerance file. The file is then used
# by diffn3.pl to determine the agreement of other test output files
# and the fiducial sample output file.
# Here is how we do regression testing. We compare test output with a
# fiducial sample output. The comparison is done only for an excerpt
# of these files. See headtail.pl for the constructions of these
# excerpts files. The diffn3.pl script works with the test excerpt
# file and fiducial excerpt file and the error tolerance file.
# This script constructs an initial error tolerance file. The file as
# constructed is somewhat speculative, since it is based on comparing
# the fiducial sample with only one test file. Further testing
# usually reveals that some of the error tolerances are too tight.
# The script trainerrfile.pl is used to relax tolerances in the error
# tolerance file to allow diffn3.pl to pass an acceptable test result.
# Compares the fiducial standard excerpt file and an acceptable test
# excerpt file. The error tolerance file is created on stdout with
# the same fields as the fiducial standard, except that numeric fields
# are replaced by whichever of the following is larger: (1) The
# absolute difference. (2) The specified tolerance (absolute) for
# magnitudes greater than 1 (3) The specified tolerance (relative) for
# magnitudes less than 1. Any nonnumeric fields that differ are
# replaced by "XXX".
# Usage...
# seterrfile.pl file1 file2 tol > errfile
# where file1 and file2 are to be compared and tol is a reasonable
# tolerance.
sub is_integer {
defined $_[0] && $_[0] =~ /^[+-]?\d+$/;
}
sub is_float {
defined $_[0] && $_[0] =~ /^[+-]?\d+(\.\d*)?$/;
}
sub is_scientific {
defined $_[0] && $_[0] =~ /^[+-]?\d+(\.\d*)?[eEdDg][+-]?(\d+)$/;
}
sub is_number {
# Allow comma or right paren after number
my $a = $_[0];
defined $a && $a =~ s/[,$\)]//;
is_integer($a) || is_float($a) || is_scientific($a);
}
($file1,$file2,$tol) = @ARGV;
(defined($tol) && defined($file2) && defined($file1)) ||
die "Usage $0 <file1> <file2> <tol>\n";
open(FILE1,$file1) || die "Couldn't open $file1: $!";
open(FILE2,$file2) || die "Couldn't open $file2: $!";
$lines = 0;
while($line1 = <FILE1>){
chop($line1);
if(!($line2 = <FILE2>))
{
die "Premature end of file on $file2\n";
}
chop($line2);
@fields1 = split(" ",$line1);
@fields2 = split(" ",$line2);
@errs = @fields1;
$i = 0;
for(@fields1)
{
# Crude test for a numeric field. Surely, we can do better.
if(is_number($_))
{
# Numeric field
if(/[^\d]/){
# Compute tolerance for noninteger
$diff = abs($_ - $fields2[$i]);
$err = $tol;
if(abs($_) > 1.){ $err = abs($tol*$_); }
if($diff > $err){ $err = $diff; }
# Round error to one sig fig
$errs[$i] = sprintf("%.1g",$err*2.0);
}
else
{
# Integers should be exact
$errs[$i] = 0;
}
}
else
{
# Nonnumeric field
if($_ ne $fields2[$i]){
$errs[$i] = "XXX";
}
}
$i++;
}
$errline = join(" ",@errs);
print "$errline\n";
$lines++;
}
if(<FILE2>)
{
die "Premature end of file on $file1\n";
}