line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Text::NumericData::App; |
2
|
|
|
|
|
|
|
|
3
|
14
|
|
|
14
|
|
4472
|
use Text::NumericData; |
|
14
|
|
|
|
|
38
|
|
|
14
|
|
|
|
|
484
|
|
4
|
14
|
|
|
14
|
|
12782
|
use Config::Param; |
|
14
|
|
|
|
|
294685
|
|
|
14
|
|
|
|
|
552
|
|
5
|
14
|
|
|
14
|
|
112
|
use Storable; |
|
14
|
|
|
|
|
32
|
|
|
14
|
|
|
|
|
799
|
|
6
|
14
|
|
|
14
|
|
104
|
use strict; |
|
14
|
|
|
|
|
41
|
|
|
14
|
|
|
|
|
20398
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
# This is just a placeholder because of a past build system bug. |
9
|
|
|
|
|
|
|
# The one and only version for Text::NumericData is kept in |
10
|
|
|
|
|
|
|
# the Text::NumericData module itself. |
11
|
|
|
|
|
|
|
our $VERSION = '1'; |
12
|
|
|
|
|
|
|
$VERSION = eval $VERSION; |
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
my %shorts = (strict=>'S', text=>'T', numformat=>'N'); |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
sub new |
17
|
|
|
|
|
|
|
{ |
18
|
19
|
|
|
19
|
0
|
70
|
my $class = shift; |
19
|
19
|
|
|
|
|
81
|
my $self = {}; |
20
|
19
|
|
|
|
|
55
|
bless $self, $class; |
21
|
19
|
|
|
|
|
100
|
$self->{setup} = shift; # main, parconf, pardef, exclude_pars |
22
|
19
|
|
|
|
|
65
|
$self->{state} = {}; |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
# safety check for misspelled keys |
25
|
19
|
|
|
|
|
147
|
my @known_setup = |
26
|
|
|
|
|
|
|
(qw( |
27
|
|
|
|
|
|
|
parconf |
28
|
|
|
|
|
|
|
pardef |
29
|
|
|
|
|
|
|
exclude |
30
|
|
|
|
|
|
|
filemode |
31
|
|
|
|
|
|
|
pipemode |
32
|
|
|
|
|
|
|
pipe_init |
33
|
|
|
|
|
|
|
pipe_file |
34
|
|
|
|
|
|
|
pipe_prefilter |
35
|
|
|
|
|
|
|
pipe_begin |
36
|
|
|
|
|
|
|
pipe_line |
37
|
|
|
|
|
|
|
pipe_end |
38
|
|
|
|
|
|
|
pipe_allend |
39
|
|
|
|
|
|
|
pipe_header |
40
|
|
|
|
|
|
|
pipe_data |
41
|
|
|
|
|
|
|
pipe_first_data |
42
|
|
|
|
|
|
|
)); |
43
|
19
|
|
|
|
|
42
|
my @unknown_keys = grep {my $k = $_; not grep {$_ eq $k} @known_setup;} (keys %{$self->{setup}}); |
|
99
|
|
|
|
|
141
|
|
|
99
|
|
|
|
|
151
|
|
|
1485
|
|
|
|
|
2366
|
|
|
19
|
|
|
|
|
100
|
|
44
|
19
|
50
|
|
|
|
90
|
print STDERR "WARNING: Text::NumericData::App got unknown setup keys (@unknown_keys)\n" if @unknown_keys; |
45
|
|
|
|
|
|
|
|
46
|
19
|
100
|
|
|
|
75
|
if($self->{setup}{pipemode}) |
47
|
|
|
|
|
|
|
{ |
48
|
|
|
|
|
|
|
# Activate pipe processing only on request. |
49
|
13
|
|
|
|
|
5541
|
require Text::ASCIIPipe; |
50
|
|
|
|
|
|
|
} |
51
|
|
|
|
|
|
|
|
52
|
19
|
50
|
|
|
|
15297
|
$self->{setup}{parconf} = {} unless defined $self->{setup}{parconf}; |
53
|
|
|
|
|
|
|
# Always return. |
54
|
19
|
|
|
|
|
72
|
$self->{setup}{parconf}{noexit} = 1; |
55
|
|
|
|
|
|
|
|
56
|
|
|
|
|
|
|
# Lazyness ... why specify this again and again? |
57
|
|
|
|
|
|
|
$self->{setup}{parconf}{copyright} = $Text::NumericData::copyright |
58
|
19
|
50
|
|
|
|
92
|
unless defined $self->{setup}{parconf}{copyright}; |
59
|
|
|
|
|
|
|
$self->{setup}{parconf}{version} = $Text::NumericData::version |
60
|
19
|
50
|
|
|
|
82
|
unless defined $self->{setup}{parconf}{version}; |
61
|
|
|
|
|
|
|
$self->{setup}{parconf}{author} = $Text::NumericData::author |
62
|
19
|
50
|
|
|
|
80
|
unless defined $self->{setup}{parconf}{author}; |
63
|
|
|
|
|
|
|
|
64
|
19
|
100
|
|
|
|
80
|
$self->{setup}{pardef} = [] unless defined $self->{setup}{pardef}; |
65
|
19
|
|
|
|
|
114
|
my $prob = Config::Param::sane_pardef($self->{setup}{parconf}, $self->{setup}{pardef}); |
66
|
19
|
50
|
|
|
|
7069
|
if($prob ne '') |
67
|
|
|
|
|
|
|
{ |
68
|
0
|
|
|
|
|
0
|
print STDERR "Error in given parameter definiton: $prob\nThis is a fatal programming error.\n"; |
69
|
0
|
|
|
|
|
0
|
return undef; |
70
|
|
|
|
|
|
|
} |
71
|
|
|
|
|
|
|
# I'm sure this can be done more elegantly. |
72
|
19
|
50
|
|
|
|
167
|
$self->add_param('Text::NumericData' |
73
|
|
|
|
|
|
|
, \%Text::NumericData::defaults, \%Text::NumericData::help) |
74
|
|
|
|
|
|
|
or return undef; |
75
|
19
|
50
|
|
|
|
94
|
$self->add_param('Text::NumericData::File' |
76
|
|
|
|
|
|
|
, \%Text::NumericData::File::defaults, \%Text::NumericData::File::help) |
77
|
|
|
|
|
|
|
or return undef; |
78
|
|
|
|
|
|
|
|
79
|
19
|
|
|
|
|
109
|
return $self; |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
sub add_param |
83
|
|
|
|
|
|
|
{ |
84
|
38
|
|
|
38
|
0
|
89
|
my $self = shift; |
85
|
38
|
|
|
|
|
103
|
my ($pkgname, $defaults, $help) = @_; |
86
|
38
|
|
|
|
|
66
|
for my $pn (keys %{$defaults}) |
|
38
|
|
|
|
|
239
|
|
87
|
|
|
|
|
|
|
{ |
88
|
270
|
50
|
33
|
|
|
655
|
next if (defined $self->{setup}{exclude} and grep {$_ eq $pn} @{$self->{setup}{exclude}}); |
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
89
|
270
|
|
|
|
|
562
|
my $help = $help->{$pn}; |
90
|
270
|
50
|
|
|
|
497
|
$help = "some $pkgname parameter" unless defined $help; |
91
|
270
|
|
|
|
|
674
|
$help .= " (from $pkgname)"; |
92
|
|
|
|
|
|
|
my $thisdef = |
93
|
|
|
|
|
|
|
{ |
94
|
|
|
|
|
|
|
long=>$pn |
95
|
|
|
|
|
|
|
, short=>$shorts{$pn} |
96
|
|
|
|
|
|
|
# No deep copy here, as the calls to Config::Param get copies of this. |
97
|
270
|
|
|
|
|
1027
|
, value=>$defaults->{$pn} |
98
|
|
|
|
|
|
|
, help=>$help |
99
|
|
|
|
|
|
|
}; |
100
|
270
|
50
|
|
|
|
719
|
if(Config::Param::sane_pardef($self->{setup}{parconf}, [$thisdef]) ne '') |
101
|
|
|
|
|
|
|
{ |
102
|
0
|
|
|
|
|
0
|
print STDERR "Unexpected failure to sanitize param definiton for $pn.\n"; |
103
|
0
|
|
|
|
|
0
|
return undef; |
104
|
|
|
|
|
|
|
} |
105
|
270
|
|
|
|
|
15951
|
push(@{$self->{setup}{pardef}}, $thisdef); |
|
270
|
|
|
|
|
695
|
|
106
|
|
|
|
|
|
|
} |
107
|
38
|
|
|
|
|
162
|
return 1; |
108
|
|
|
|
|
|
|
} |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
sub run |
111
|
|
|
|
|
|
|
{ |
112
|
46
|
|
|
46
|
1
|
36039
|
my ($self, $argv, $in, $out) = @_; |
113
|
46
|
50
|
|
|
|
153
|
$argv = \@ARGV unless defined $argv; |
114
|
46
|
100
|
|
|
|
124
|
$in = \*STDIN unless defined $in; |
115
|
46
|
50
|
|
|
|
125
|
$out = \*STDOUT unless defined $out; |
116
|
46
|
|
|
|
|
128
|
binmode $in; |
117
|
46
|
|
|
|
|
83
|
binmode $out; |
118
|
46
|
|
|
|
|
118
|
$self->{argv} = $argv; |
119
|
46
|
|
|
|
|
107
|
$self->{in} = $in; |
120
|
46
|
|
|
|
|
120
|
$self->{out} = $out; |
121
|
|
|
|
|
|
|
|
122
|
46
|
|
|
|
|
71
|
my $errors; |
123
|
|
|
|
|
|
|
# Ensure that Config::Param cannot mess with our default values by |
124
|
|
|
|
|
|
|
# providing deep copies of configuration. Even if it behaves nice, |
125
|
|
|
|
|
|
|
# better safe than sorry. |
126
|
|
|
|
|
|
|
$self->{param} = Config::Param::get( |
127
|
|
|
|
|
|
|
Storable::dclone($self->{setup}{parconf}) |
128
|
|
|
|
|
|
|
, Storable::dclone($self->{setup}{pardef}) |
129
|
46
|
|
|
|
|
6724
|
, $self->{argv}, $errors ); |
130
|
|
|
|
|
|
|
|
131
|
46
|
50
|
|
|
|
367013
|
if(@{$errors}) |
|
46
|
|
|
|
|
253
|
|
132
|
|
|
|
|
|
|
{ |
133
|
0
|
|
|
|
|
0
|
print STDERR "Stopping here because of parameter parsing errors.\n"; |
134
|
0
|
|
|
|
|
0
|
return 1; |
135
|
|
|
|
|
|
|
} |
136
|
46
|
50
|
33
|
|
|
325
|
return 0 if($self->{param}{help} or $self->{param}{version}); |
137
|
|
|
|
|
|
|
|
138
|
46
|
100
|
|
|
|
147
|
if($self->{setup}{pipemode}) |
139
|
|
|
|
|
|
|
{ |
140
|
39
|
100
|
|
|
|
121
|
if(defined $self->{setup}{pipe_init}) |
141
|
|
|
|
|
|
|
{ |
142
|
35
|
|
|
|
|
161
|
my $err = $self->{setup}{pipe_init}->($self); |
143
|
35
|
50
|
|
|
|
155
|
if($err) |
144
|
|
|
|
|
|
|
{ |
145
|
0
|
|
|
|
|
0
|
print STDERR "Pipe init handler failed, aborting.\n"; |
146
|
0
|
|
|
|
|
0
|
return $err; |
147
|
|
|
|
|
|
|
} |
148
|
|
|
|
|
|
|
} |
149
|
39
|
100
|
|
|
|
120
|
if($self->{setup}{filemode}) |
150
|
|
|
|
|
|
|
{ |
151
|
|
|
|
|
|
|
# Not wholly sure about that logic, have to really test and then delete this comment. |
152
|
11
|
|
|
|
|
29
|
while(1) |
153
|
|
|
|
|
|
|
{ |
154
|
11
|
|
|
|
|
63
|
$self->new_txd(); |
155
|
11
|
|
|
|
|
52
|
my $ret = $self->{txd}->read_all($self->{in}); |
156
|
|
|
|
|
|
|
$self->{setup}{pipe_file}->($self) |
157
|
11
|
50
|
33
|
|
|
138
|
if ($ret >= 0 and defined $self->{setup}{pipe_file}); |
158
|
11
|
50
|
|
|
|
45
|
last if $ret <= 0; |
159
|
|
|
|
|
|
|
} |
160
|
|
|
|
|
|
|
$self->{setup}{pipe_allend}->($self) |
161
|
11
|
50
|
|
|
|
59
|
if defined $self->{setup}{pipe_allend}; |
162
|
|
|
|
|
|
|
Text::ASCIIPipe::done($self->{out}) |
163
|
11
|
50
|
|
|
|
46
|
if $self->{txd}{config}{pipemode}; |
164
|
|
|
|
|
|
|
} |
165
|
|
|
|
|
|
|
else |
166
|
|
|
|
|
|
|
{ |
167
|
|
|
|
|
|
|
Text::ASCIIPipe::process |
168
|
|
|
|
|
|
|
( |
169
|
|
|
|
|
|
|
handle => $self |
170
|
|
|
|
|
|
|
,in => $self->{in} |
171
|
|
|
|
|
|
|
,out => $self->{out} |
172
|
|
|
|
|
|
|
,pre => $self->{setup}{pipe_prefilter} |
173
|
|
|
|
|
|
|
,begin => defined $self->{setup}{pipe_begin} ? $self->{setup}{pipe_begin} : \&new_txd |
174
|
|
|
|
|
|
|
,line => defined $self->{setup}{pipe_line} ? $self->{setup}{pipe_line} : \&default_line_hook |
175
|
|
|
|
|
|
|
,end => $self->{setup}{pipe_end} |
176
|
|
|
|
|
|
|
,allend => $self->{setup}{pipe_allend} |
177
|
28
|
50
|
|
|
|
309
|
); |
|
|
100
|
|
|
|
|
|
178
|
|
|
|
|
|
|
} |
179
|
39
|
|
|
|
|
1337
|
return 0; |
180
|
|
|
|
|
|
|
} |
181
|
|
|
|
|
|
|
else |
182
|
|
|
|
|
|
|
{ |
183
|
7
|
|
|
|
|
44
|
$self->new_txd(); |
184
|
7
|
|
|
|
|
40
|
return $self->main(); |
185
|
|
|
|
|
|
|
} |
186
|
|
|
|
|
|
|
} |
187
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
sub new_txd |
189
|
|
|
|
|
|
|
{ |
190
|
44
|
|
|
44
|
1
|
95
|
my $self = shift; |
191
|
44
|
100
|
|
|
|
2541
|
require Text::NumericData::File if $self->{setup}{filemode}; |
192
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
$self->{txd} = $self->{setup}{filemode} |
194
|
|
|
|
|
|
|
? Text::NumericData::File->new($self->{param}) |
195
|
44
|
100
|
|
|
|
354
|
: Text::NumericData->new($self->{param}); |
196
|
|
|
|
|
|
|
|
197
|
44
|
|
|
|
|
151
|
$self->{state}{data} = 0; |
198
|
|
|
|
|
|
|
} |
199
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
|
201
|
|
|
|
|
|
|
sub default_line_hook |
202
|
|
|
|
|
|
|
{ |
203
|
492
|
|
|
492
|
1
|
14866
|
my $self = shift; |
204
|
492
|
|
|
|
|
718
|
my $prefix = undef; |
205
|
492
|
100
|
|
|
|
1188
|
if(!$self->{state}{data}) |
206
|
|
|
|
|
|
|
{ |
207
|
42
|
100
|
|
|
|
139
|
if($self->{txd}->line_check($_[0])) |
208
|
|
|
|
|
|
|
{ |
209
|
18
|
|
|
|
|
50
|
$self->{state}{data} = 1; |
210
|
|
|
|
|
|
|
$prefix = $self->{setup}{pipe_first_data}->($self, @_) |
211
|
18
|
100
|
|
|
|
64
|
if defined $self->{setup}{pipe_first_data}; |
212
|
|
|
|
|
|
|
} |
213
|
|
|
|
|
|
|
else |
214
|
|
|
|
|
|
|
{ |
215
|
|
|
|
|
|
|
$self->{setup}{pipe_header}->($self, @_) |
216
|
24
|
50
|
|
|
|
124
|
if defined $self->{setup}{pipe_header}; |
217
|
24
|
|
|
|
|
53
|
return; |
218
|
|
|
|
|
|
|
} |
219
|
|
|
|
|
|
|
} |
220
|
|
|
|
|
|
|
# If still here, $self->{state}{data} == 1 is implied. |
221
|
|
|
|
|
|
|
$self->{setup}{pipe_data}->($self, @_) |
222
|
468
|
50
|
|
|
|
1862
|
if defined $self->{setup}{pipe_data}; |
223
|
468
|
100
|
|
|
|
1566
|
$_[0] = ${$prefix}.$_[0] |
|
7
|
|
|
|
|
29
|
|
224
|
|
|
|
|
|
|
if defined $prefix; |
225
|
|
|
|
|
|
|
} |
226
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
sub error |
228
|
|
|
|
|
|
|
{ |
229
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
230
|
0
|
0
|
|
|
|
|
print STDERR "$_[0]\n" if defined $_[0]; |
231
|
0
|
0
|
|
|
|
|
return defined $_[1] ? $_[1] : -1; |
232
|
|
|
|
|
|
|
} |
233
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
1; |
235
|
|
|
|
|
|
|
|
236
|
|
|
|
|
|
|
__END__ |