| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Log::ProgramInfo; |
|
2
|
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
=head1 NAME |
|
4
|
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
Log::ProgramInfo - log global info from a perl programs. |
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=cut |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
### HISTORY ################################################################### |
|
10
|
|
|
|
|
|
|
# Version Date Developer Comments |
|
11
|
|
|
|
|
|
|
# 0.1.1 2015-04-02 John Macdonald Initial release to CPAN |
|
12
|
|
|
|
|
|
|
# 0.1.2 2015-04-04 John Macdonald Minor cleanups to initial release |
|
13
|
|
|
|
|
|
|
# 0.1.3 2015-04-09 John Macdonald Rename s/JobInof/ProgramInfo/ |
|
14
|
|
|
|
|
|
|
# 0.1.4 2015-04-09 John Macdonald Add README |
|
15
|
|
|
|
|
|
|
# 0.1.5 2015-04-10 John Macdonald Change log extension and env |
|
16
|
|
|
|
|
|
|
# variable names to reflect |
|
17
|
|
|
|
|
|
|
# the ProgramInfo name. |
|
18
|
|
|
|
|
|
|
# 0.1.6 2015-04-23 John Macdonald Fix env name ..._FILE -> ..._NAME |
|
19
|
|
|
|
|
|
|
# Fix env handling - capture at start |
|
20
|
|
|
|
|
|
|
# - apply at end |
|
21
|
|
|
|
|
|
|
# Log username(s) groupname(s) |
|
22
|
|
|
|
|
|
|
# Include identifying info in separator line |
|
23
|
|
|
|
|
|
|
# Make output more easily parsable |
|
24
|
|
|
|
|
|
|
# 0.1.7 2015-05-20 John Macdonald Add user-pluggable log routines |
|
25
|
|
|
|
|
|
|
# Move log parser from test to module. |
|
26
|
|
|
|
|
|
|
# 0.1.8 2015-06-12 John Macdonald Catch most signals (not just HUP) |
|
27
|
|
|
|
|
|
|
# |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
=head1 VERSION |
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
Version 0.1.8 |
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
=cut |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
our $VERSION = '0.1.8'; |
|
36
|
|
|
|
|
|
|
|
|
37
|
1
|
|
|
1
|
|
28680
|
use feature qw(say); |
|
|
1
|
|
|
|
|
4
|
|
|
|
1
|
|
|
|
|
128
|
|
|
38
|
1
|
|
|
1
|
|
829
|
use Data::Dumper; |
|
|
1
|
|
|
|
|
10879
|
|
|
|
1
|
|
|
|
|
100
|
|
|
39
|
1
|
|
|
1
|
|
635
|
use FindBin; |
|
|
1
|
|
|
|
|
1134
|
|
|
|
1
|
|
|
|
|
49
|
|
|
40
|
1
|
|
|
1
|
|
724
|
use Time::HiRes qw(time); |
|
|
1
|
|
|
|
|
1901
|
|
|
|
1
|
|
|
|
|
5
|
|
|
41
|
1
|
|
|
1
|
|
1208
|
use DateTime; |
|
|
1
|
|
|
|
|
160626
|
|
|
|
1
|
|
|
|
|
56
|
|
|
42
|
1
|
|
|
1
|
|
13
|
use DateTime::Duration; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
35
|
|
|
43
|
1
|
|
|
1
|
|
6
|
use Carp qw(carp croak cluck); |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
105
|
|
|
44
|
1
|
|
|
1
|
|
8
|
use Fcntl qw(:flock); |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
1643
|
|
|
45
|
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
48
|
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
use Log::ProgramInfo qw( |
|
50
|
|
|
|
|
|
|
[ -logname LOGNAME ] |
|
51
|
|
|
|
|
|
|
[ -logdir LOGDIR ] |
|
52
|
|
|
|
|
|
|
[ -logext LOGEXT ] |
|
53
|
|
|
|
|
|
|
[ -logdate none|date|time|datetime ] |
|
54
|
|
|
|
|
|
|
[ -stdout ] |
|
55
|
|
|
|
|
|
|
[ -suppress ] |
|
56
|
|
|
|
|
|
|
); |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
# main program does lots of stuff... |
|
59
|
|
|
|
|
|
|
exit 0; |
|
60
|
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
After the program has run, this module will automatically |
|
62
|
|
|
|
|
|
|
log information about this run into a log file. It will |
|
63
|
|
|
|
|
|
|
list such things as: |
|
64
|
|
|
|
|
|
|
- program |
|
65
|
|
|
|
|
|
|
- name |
|
66
|
|
|
|
|
|
|
- version |
|
67
|
|
|
|
|
|
|
- command line arguments |
|
68
|
|
|
|
|
|
|
- version of perl |
|
69
|
|
|
|
|
|
|
- modules loaded |
|
70
|
|
|
|
|
|
|
- source code location |
|
71
|
|
|
|
|
|
|
- Version |
|
72
|
|
|
|
|
|
|
- run time |
|
73
|
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
The log is appended to the file: |
|
75
|
|
|
|
|
|
|
LOGDIR/LOGDATE-LOGNAME.LOGEXT |
|
76
|
|
|
|
|
|
|
where |
|
77
|
|
|
|
|
|
|
LOGDIR defaults to . (the current directory when the program terminates) |
|
78
|
|
|
|
|
|
|
LOGDATE defaults to the date that the program was started |
|
79
|
|
|
|
|
|
|
LOGNAME defaults to the basename of the program |
|
80
|
|
|
|
|
|
|
LOGEXT defaults to ".programinfo" |
|
81
|
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
The -ARG specifiers in the "import" list can be used to over-ride these defaults. Specifying: |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
-logname LOGNAME will use LOGNAME instead of the program name |
|
85
|
|
|
|
|
|
|
-logdir LOGDIR will use LOGDIR instead of the current directory |
|
86
|
|
|
|
|
|
|
- if it is a relative path, it will be based on the |
|
87
|
|
|
|
|
|
|
current directory at termination of execution |
|
88
|
|
|
|
|
|
|
-logext EXT will add .EXT to the log filename |
|
89
|
|
|
|
|
|
|
-logext .EXT will add .EXT to the log filename |
|
90
|
|
|
|
|
|
|
-logext "" will add no extension to the log filename |
|
91
|
|
|
|
|
|
|
-logdate STRING |
|
92
|
|
|
|
|
|
|
will specify the LOGDATE portion of the filename. The STRING can be: |
|
93
|
|
|
|
|
|
|
none LOGNAME (and no dash) |
|
94
|
|
|
|
|
|
|
date YYYYMMDD-LOGNAME (this is the default) |
|
95
|
|
|
|
|
|
|
time HHMMSS-LOGNAME |
|
96
|
|
|
|
|
|
|
datetime YYYYMMDDHHMMSS-LOGNAME |
|
97
|
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
-stdout will cause the log to be sent to stdout instead of a file |
|
99
|
|
|
|
|
|
|
-suppress will suppress logging (unless environment variable |
|
100
|
|
|
|
|
|
|
LOGPROGRAMINFO_SUPPRESS is explcitly set to 0 or null) |
|
101
|
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
Normally, neither -suppress nor -stdout will be set in the |
|
103
|
|
|
|
|
|
|
use statement, and the environment variables can then be |
|
104
|
|
|
|
|
|
|
used to disable the logging completely or to send it to |
|
105
|
|
|
|
|
|
|
stdout instead of to the selected file. |
|
106
|
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
For some programs, however, it may be desired to not normally |
|
108
|
|
|
|
|
|
|
provide any logging. Specifying -suppress will accomplish |
|
109
|
|
|
|
|
|
|
this. In such a case, setting the environment variable |
|
110
|
|
|
|
|
|
|
LOGPROGRAMINFO_SUPPRESS=0 will over-ride that choice, causing |
|
111
|
|
|
|
|
|
|
the log to be written (as specified by the other options |
|
112
|
|
|
|
|
|
|
and environment variables). |
|
113
|
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
Environment variables can over-ride these parameters: |
|
115
|
|
|
|
|
|
|
LOGPROGRAMINFO_SUPPRESS=x boolean suppresses all logging if true |
|
116
|
|
|
|
|
|
|
LOGPROGRAMINFO_STDOUT=x boolean sets -stdout |
|
117
|
|
|
|
|
|
|
LOGPROGRAMINFO_DIR=DIR string sets the target directory name |
|
118
|
|
|
|
|
|
|
LOGPROGRAMINFO_NAME=NAME string sets the target filename LOGNAME |
|
119
|
|
|
|
|
|
|
LOGPROGRAMINFO_EXT=EXT string sets the target extension |
|
120
|
|
|
|
|
|
|
LOGPROGRAMINFO_DATE=DATE string sets the target filename LOGDATE selector |
|
121
|
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
Adding extra loggable information: |
|
123
|
|
|
|
|
|
|
If you want to add your own classes of loggable info, there are a |
|
124
|
|
|
|
|
|
|
few restrictions. |
|
125
|
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
You define a logging extension routine using: |
|
127
|
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
Log::ProgramInfo::add_extra_logger( \&my_logger ); |
|
129
|
|
|
|
|
|
|
|
|
130
|
|
|
|
|
|
|
Your logger routine should be defined as: |
|
131
|
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
sub my_logger { |
|
133
|
|
|
|
|
|
|
my $write_entry = shift; |
|
134
|
|
|
|
|
|
|
$write_entry->( $key1, $value ); |
|
135
|
|
|
|
|
|
|
$write_entry->( $key1, $key2, $value ); |
|
136
|
|
|
|
|
|
|
} |
|
137
|
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
The $write_entry function passed to my_logger must be called with |
|
139
|
|
|
|
|
|
|
2 or 3 arguments. The leading arguments are major (and minor if |
|
140
|
|
|
|
|
|
|
desired) keys, the final one os the value for the key(s). |
|
141
|
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
Try to keep the first key to 7 characters, and the second to 8 to |
|
143
|
|
|
|
|
|
|
keep the log readable by humans. (It will be parseable even if you |
|
144
|
|
|
|
|
|
|
use longer keys.) |
|
145
|
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
Parsing the log file: |
|
147
|
|
|
|
|
|
|
The log file is designed to be easily parsed. |
|
148
|
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
A log always starts with a line beginning with 8 hash marks in column |
|
150
|
|
|
|
|
|
|
one (########) plus some identifying info. |
|
151
|
|
|
|
|
|
|
|
|
152
|
|
|
|
|
|
|
The value lines are of the form: |
|
153
|
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
key : value |
|
155
|
|
|
|
|
|
|
key1 : key2 : value |
|
156
|
|
|
|
|
|
|
|
|
157
|
|
|
|
|
|
|
The first key is extended to at least 7 characters with blanks, the |
|
158
|
|
|
|
|
|
|
second key (if any) is extended to at least 8 characters. There is |
|
159
|
|
|
|
|
|
|
always a separator of (space(colon)(space) between a key and the |
|
160
|
|
|
|
|
|
|
following field. (A key can be provided with leading spaces for making |
|
161
|
|
|
|
|
|
|
the log more readable by humans - the readlog function in the test suite |
|
162
|
|
|
|
|
|
|
will remove such spaces.) |
|
163
|
|
|
|
|
|
|
|
|
164
|
|
|
|
|
|
|
Two subroutines are available to do this parsing for you: |
|
165
|
|
|
|
|
|
|
|
|
166
|
|
|
|
|
|
|
my $firstonly = 0; |
|
167
|
|
|
|
|
|
|
@logs = readlog( $filepath [, $acceptsub] [, $firstonly] ); |
|
168
|
|
|
|
|
|
|
@logs = parselog( $filehandle [, $acceptsub] [, $firstonly] ); |
|
169
|
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
$logs = readlog( $filepath [, $acceptsub ], 1 ); |
|
171
|
|
|
|
|
|
|
$logs = parselog( $filehandle [, $acceptsub ] ,1 ); |
|
172
|
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
The first parameter to each is either a string pathname (for readlog) |
|
174
|
|
|
|
|
|
|
or an already opened readable file handle (for parselog). |
|
175
|
|
|
|
|
|
|
|
|
176
|
|
|
|
|
|
|
If a subroutine reference arg $acceptsub is provided, each log that is |
|
177
|
|
|
|
|
|
|
read will be passed to that sub reference. If the acceptsub returns |
|
178
|
|
|
|
|
|
|
true the log is retained, otherwise it is discarded. If a trailing |
|
179
|
|
|
|
|
|
|
(non-sub-ref) value is provided, it selects whether only the first |
|
180
|
|
|
|
|
|
|
(acceptable) log found will be returned as a single hash reference, or |
|
181
|
|
|
|
|
|
|
whether all of the (accepted) logs in the file will be returned as a |
|
182
|
|
|
|
|
|
|
list of hash references.` |
|
183
|
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
The hash reference for each accepted log contains the key/value (or |
|
185
|
|
|
|
|
|
|
key1 => { key2/value pairs }) from that log. |
|
186
|
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
Whenever a key (or key1/key2 pair) is seen multiple times, the value |
|
188
|
|
|
|
|
|
|
is an array ref instead of a scalar. This only happens with the |
|
189
|
|
|
|
|
|
|
MODULE pairs (MODULE/NAME, MODULE/LOC, MODULE/VERSION), and the INC |
|
190
|
|
|
|
|
|
|
key. (User-provided keys are not currently permitted to use the same |
|
191
|
|
|
|
|
|
|
key set multiple times.) |
|
192
|
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
=cut |
|
194
|
|
|
|
|
|
|
|
|
195
|
|
|
|
|
|
|
# preserve command line info |
|
196
|
|
|
|
|
|
|
my @args = @ARGV; |
|
197
|
|
|
|
|
|
|
my $progbase; |
|
198
|
|
|
|
|
|
|
my $starttime; |
|
199
|
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
my %option; |
|
201
|
|
|
|
|
|
|
|
|
202
|
|
|
|
|
|
|
my %valid_dates; |
|
203
|
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
my %_omap; |
|
205
|
|
|
|
|
|
|
my %env_options; |
|
206
|
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
my $kill_caught; |
|
208
|
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
my ($uid, $gid); |
|
210
|
|
|
|
|
|
|
my %cache; |
|
211
|
|
|
|
|
|
|
my %modkeys = ( NAME => 1, VERSION => 1, LOC => 1 ); |
|
212
|
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
sub readlog { |
|
214
|
4
|
|
|
4
|
0
|
10650619
|
my $file = shift; |
|
215
|
4
|
50
|
|
|
|
221
|
open my $fh, '<', $file or die "Cannot open $file: $!"; |
|
216
|
4
|
|
|
|
|
35
|
return parselog( $fh, @_ ); |
|
217
|
|
|
|
|
|
|
} |
|
218
|
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
sub parselog { |
|
220
|
4
|
|
|
4
|
0
|
48
|
my $fh = shift; |
|
221
|
4
|
|
|
|
|
10
|
my $accept; |
|
222
|
4
|
50
|
|
|
|
19
|
$accept = shift if ref($_[0]) eq 'CODE'; |
|
223
|
4
|
|
|
|
|
11
|
my $firstonly = shift; |
|
224
|
4
|
|
|
|
|
8
|
my @logs; |
|
225
|
4
|
|
|
|
|
18
|
while (my $log = parse1log( $fh )) { |
|
226
|
4
|
50
|
33
|
|
|
14
|
next if $accept && ! $accept->($log); |
|
227
|
4
|
50
|
|
|
|
10
|
return $log if $firstonly; |
|
228
|
4
|
|
|
|
|
19
|
push @logs, $log; |
|
229
|
|
|
|
|
|
|
} |
|
230
|
4
|
|
|
|
|
233
|
return @logs; |
|
231
|
|
|
|
|
|
|
} |
|
232
|
|
|
|
|
|
|
|
|
233
|
|
|
|
|
|
|
sub parse1log { |
|
234
|
8
|
|
|
8
|
0
|
14
|
my $fh = shift; |
|
235
|
8
|
|
|
|
|
15
|
my $log; |
|
236
|
8
|
|
|
|
|
213
|
while (my $line = <$fh>) { |
|
237
|
827
|
100
|
|
|
|
2058
|
return $log if $line =~ /^$/; |
|
238
|
823
|
100
|
|
|
|
1507
|
next if $line =~ /^########/; |
|
239
|
819
|
|
|
|
|
870
|
chomp $line; |
|
240
|
819
|
|
100
|
|
|
1312
|
$log ||= {}; |
|
241
|
819
|
|
|
|
|
2296
|
my @keys = split ': ', $line; |
|
242
|
819
|
|
|
|
|
5185
|
s/^\s*// for @keys; |
|
243
|
819
|
|
|
|
|
7692
|
s/\s*$// for @keys; |
|
244
|
819
|
50
|
|
|
|
1720
|
die "Unexpected syntax in log line: $line\n" unless scalar(@keys) >= 2; |
|
245
|
819
|
|
|
|
|
1034
|
my $val = pop @keys; |
|
246
|
819
|
|
|
|
|
1059
|
my $key = shift @keys; |
|
247
|
819
|
100
|
|
|
|
1190
|
if (scalar(@keys) == 0) { |
|
248
|
77
|
100
|
|
|
|
108
|
if ($key eq 'INC') { |
|
249
|
28
|
|
100
|
|
|
108
|
my $list = $log->{$key} ||= []; |
|
250
|
28
|
|
|
|
|
113
|
push @$list, $val; |
|
251
|
|
|
|
|
|
|
} |
|
252
|
|
|
|
|
|
|
else { |
|
253
|
49
|
50
|
|
|
|
100
|
die "repeated key: $key" if exists $log->{$key}; |
|
254
|
49
|
|
|
|
|
334
|
$log->{$key} = $val; |
|
255
|
|
|
|
|
|
|
} |
|
256
|
|
|
|
|
|
|
} |
|
257
|
|
|
|
|
|
|
else { |
|
258
|
742
|
|
|
|
|
820
|
my $key2 = shift @keys; |
|
259
|
742
|
50
|
|
|
|
1298
|
die "invalid nested key: {" . join( '}{', $key, $key2, @keys, $val ) . "}" |
|
260
|
|
|
|
|
|
|
if scalar(@keys); |
|
261
|
742
|
100
|
|
|
|
1483
|
if ($key eq 'MODULE') { |
|
262
|
720
|
50
|
|
|
|
1470
|
die "Unknown MODULE key ($key2)" unless $modkeys{$key2}; |
|
263
|
720
|
|
100
|
|
|
1692
|
my $list = $log->{$key}{$key2} ||= []; |
|
264
|
720
|
|
|
|
|
3035
|
push @$list, $val; |
|
265
|
|
|
|
|
|
|
} |
|
266
|
|
|
|
|
|
|
else { |
|
267
|
22
|
50
|
|
|
|
76
|
die "repeated key: $key $key2" if exists $log->{$key}{$key2}; |
|
268
|
22
|
|
|
|
|
139
|
$log->{$key}{$key2} = $val; |
|
269
|
|
|
|
|
|
|
} |
|
270
|
|
|
|
|
|
|
} |
|
271
|
|
|
|
|
|
|
} |
|
272
|
4
|
|
|
|
|
13
|
return $log; |
|
273
|
|
|
|
|
|
|
} |
|
274
|
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
my @extra_loggers = (); |
|
276
|
|
|
|
|
|
|
|
|
277
|
|
|
|
|
|
|
sub add_extra_logger { |
|
278
|
1
|
|
|
1
|
0
|
3045
|
for my $logger (@_) { |
|
279
|
1
|
50
|
|
|
|
12
|
croak "arg to extra_loggers is not a code ref: " . Dumper($logger) |
|
280
|
|
|
|
|
|
|
unless ref $logger eq 'CODE'; |
|
281
|
0
|
|
|
|
|
0
|
push @extra_loggers, $logger; |
|
282
|
|
|
|
|
|
|
} |
|
283
|
|
|
|
|
|
|
} |
|
284
|
|
|
|
|
|
|
|
|
285
|
|
|
|
|
|
|
sub groupmap { |
|
286
|
2
|
|
|
2
|
0
|
10
|
my $list = shift; |
|
287
|
2
|
|
|
|
|
3
|
my @res; |
|
288
|
|
|
|
|
|
|
my %unique; |
|
289
|
2
|
|
66
|
|
|
7
|
push @res, ($cache{$_} //= getgrgid $_) for grep { ! $unique{$_}++ } split ' ', $list; |
|
|
2
|
|
|
|
|
124
|
|
|
290
|
2
|
|
|
|
|
5
|
my $g1 = shift @res; |
|
291
|
2
|
|
|
|
|
8
|
return join( '+', $g1, join( ',', @res ) ); |
|
292
|
|
|
|
|
|
|
} |
|
293
|
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
BEGIN { |
|
295
|
1
|
|
|
1
|
|
3
|
$progbase = $FindBin::Script; |
|
296
|
1
|
|
|
|
|
9
|
$starttime = DateTime->from_epoch(epoch => time); |
|
297
|
1
|
|
|
|
|
394
|
$valid_dates{$_} = 1 for qw( date time datetime none ); |
|
298
|
1
|
|
|
|
|
741
|
$uid = getpwuid $<; |
|
299
|
1
|
|
|
|
|
39
|
my $euid = getpwuid $>; |
|
300
|
1
|
|
|
|
|
5
|
$gid = groupmap $(; |
|
301
|
1
|
|
|
|
|
3
|
my $egid = groupmap $); |
|
302
|
1
|
50
|
|
|
|
5
|
$uid = "$euid($uid)" if $uid ne $euid; |
|
303
|
1
|
50
|
|
|
|
8
|
$gid = "$egid // $gid" if $egid ne $gid; |
|
304
|
|
|
|
|
|
|
|
|
305
|
1
|
|
|
|
|
21
|
%option = ( |
|
306
|
|
|
|
|
|
|
suppress => 0, |
|
307
|
|
|
|
|
|
|
stdout => 0, |
|
308
|
|
|
|
|
|
|
logdir => ".", |
|
309
|
|
|
|
|
|
|
logdate => "date", |
|
310
|
|
|
|
|
|
|
logname => $progbase, |
|
311
|
|
|
|
|
|
|
logext => ".programinfo", |
|
312
|
|
|
|
|
|
|
); |
|
313
|
|
|
|
|
|
|
|
|
314
|
1
|
|
|
|
|
7
|
%_omap = ( |
|
315
|
|
|
|
|
|
|
LOGPROGRAMINFO_SUPPRESS => 'suppress', |
|
316
|
|
|
|
|
|
|
LOGPROGRAMINFO_STDOUT => 'stdout', |
|
317
|
|
|
|
|
|
|
LOGPROGRAMINFO_DIR => 'logdir', |
|
318
|
|
|
|
|
|
|
LOGPROGRAMINFO_DATE => 'logdate', |
|
319
|
|
|
|
|
|
|
LOGPROGRAMINFO_NAME => 'logname', |
|
320
|
|
|
|
|
|
|
LOGPROGRAMINFO_EXT => 'logext', |
|
321
|
|
|
|
|
|
|
); |
|
322
|
|
|
|
|
|
|
|
|
323
|
1
|
|
|
|
|
11
|
while( my($k,$v) = each %_omap ) { |
|
324
|
6
|
50
|
|
|
|
26
|
$env_options{$v} = $ENV{$k} if exists $ENV{$k}; |
|
325
|
|
|
|
|
|
|
} |
|
326
|
1
|
|
|
|
|
16
|
$SIG{HUP} = \&catch_sig; |
|
327
|
1
|
|
|
|
|
6
|
$SIG{INT} = \&catch_sig; |
|
328
|
1
|
|
|
|
|
4
|
$SIG{PIPE} = \&catch_sig; |
|
329
|
1
|
|
|
|
|
13
|
$SIG{TERM} = \&catch_sig; |
|
330
|
1
|
|
|
|
|
5
|
$SIG{USR1} = \&catch_sig; |
|
331
|
1
|
|
|
|
|
2041
|
$SIG{USR2} = \&catch_sig; |
|
332
|
|
|
|
|
|
|
} |
|
333
|
|
|
|
|
|
|
|
|
334
|
|
|
|
|
|
|
sub import { |
|
335
|
1
|
|
|
1
|
|
16
|
my $mod = shift; |
|
336
|
|
|
|
|
|
|
|
|
337
|
1
|
|
|
|
|
4
|
while (scalar(@_)) { |
|
338
|
1
|
50
|
|
|
|
13
|
if ($_[0] =~ /^-(logname|logdir|logext|logdate)$/) { |
|
|
|
0
|
|
|
|
|
|
|
339
|
1
|
|
|
|
|
4
|
my $key = $1; |
|
340
|
1
|
50
|
|
|
|
3
|
croak "Option to Log::ProgramInfo requires a value: $_[0]" if scalar(@_) == 1; |
|
341
|
1
|
|
|
|
|
2
|
shift; |
|
342
|
1
|
|
|
|
|
3
|
my $val = shift; |
|
343
|
1
|
|
|
|
|
5
|
$option{$key} = $val; |
|
344
|
|
|
|
|
|
|
} |
|
345
|
|
|
|
|
|
|
elsif ($_[0] =~ /^-(stdout|suppress)$/) { |
|
346
|
0
|
|
|
|
|
0
|
my $key = $1; |
|
347
|
0
|
|
|
|
|
0
|
shift; |
|
348
|
0
|
|
|
|
|
0
|
$option{$key} = 1; |
|
349
|
|
|
|
|
|
|
} |
|
350
|
|
|
|
|
|
|
else { |
|
351
|
0
|
|
|
|
|
0
|
last; |
|
352
|
|
|
|
|
|
|
} |
|
353
|
|
|
|
|
|
|
} |
|
354
|
|
|
|
|
|
|
|
|
355
|
1
|
50
|
33
|
|
|
6
|
croak "Unknown option to Log::ProgramInfo: $_[0]" if (@_ and $_[0] =~ /^-/); |
|
356
|
1
|
50
|
|
|
|
3
|
croak "Import arguments not supported from Log::ProgramInfo: " . join( ',', @_ ) if @_; |
|
357
|
1
|
50
|
|
|
|
4
|
croak "Unknown logdate option: $option{logdate}" |
|
358
|
|
|
|
|
|
|
unless exists $valid_dates{ $option{logdate} }; |
|
359
|
|
|
|
|
|
|
|
|
360
|
1
|
50
|
|
|
|
20
|
say STDERR "resolved option hash: ", Dumper(\%option) if $ENV{DUMP_LOG_IMPORTS}; |
|
361
|
|
|
|
|
|
|
} |
|
362
|
|
|
|
|
|
|
|
|
363
|
|
|
|
|
|
|
END { |
|
364
|
1
|
|
|
1
|
|
7443
|
my $exit_status = $?; |
|
365
|
1
|
|
|
|
|
3
|
local $?; # protect program exit code from END actions |
|
366
|
1
|
|
|
|
|
5
|
finish_log($exit_status); |
|
367
|
|
|
|
|
|
|
} |
|
368
|
|
|
|
|
|
|
|
|
369
|
|
|
|
|
|
|
sub catch_sig { |
|
370
|
0
|
|
|
0
|
0
|
0
|
my $signame = shift; |
|
371
|
0
|
|
|
|
|
0
|
local $?; # protect program exit code from END actions |
|
372
|
0
|
|
|
|
|
0
|
finish_log("Killed with signal: $signame"); |
|
373
|
|
|
|
|
|
|
} |
|
374
|
|
|
|
|
|
|
|
|
375
|
|
|
|
|
|
|
sub log_entry { |
|
376
|
355
|
|
|
355
|
0
|
355
|
my $logfh = shift; |
|
377
|
355
|
100
|
|
|
|
522
|
if (@_ == 2 ) { |
|
|
|
50
|
|
|
|
|
|
|
378
|
23
|
|
|
|
|
62
|
printf $logfh "%-7s : %s\n", @_; |
|
379
|
|
|
|
|
|
|
} elsif (@_ == 3 ) { |
|
380
|
332
|
|
|
|
|
812
|
printf $logfh "%-7s : %-8s : %s\n", @_; |
|
381
|
|
|
|
|
|
|
} else { |
|
382
|
0
|
|
|
|
|
0
|
my $msg = "log_entry needs 2 or 3 arguments, got " |
|
383
|
|
|
|
|
|
|
. scalar(@_); |
|
384
|
0
|
0
|
|
|
|
0
|
$msg .= ': (' . join( '), (', @_ ) . ')' if @_; |
|
385
|
0
|
|
|
|
|
0
|
cluck $msg; |
|
386
|
|
|
|
|
|
|
} |
|
387
|
|
|
|
|
|
|
} |
|
388
|
|
|
|
|
|
|
|
|
389
|
|
|
|
|
|
|
sub finish_log { |
|
390
|
1
|
50
|
|
1
|
0
|
5
|
return if $kill_caught++; # only write log once - first kill, or termination |
|
391
|
1
|
|
|
|
|
2
|
my $exit_status = shift; |
|
392
|
1
|
50
|
|
|
|
6
|
unless ($option{suppress}) { |
|
393
|
|
|
|
|
|
|
# pull ENV var over-rides |
|
394
|
1
|
|
|
|
|
7
|
while (my ($k, $v) = each %env_options) { |
|
395
|
0
|
|
|
|
|
0
|
$option{$k} = $v; |
|
396
|
|
|
|
|
|
|
} |
|
397
|
1
|
|
|
|
|
2
|
my $logfh; |
|
398
|
1
|
|
|
|
|
15
|
my $endtime = DateTime->from_epoch(epoch => time); |
|
399
|
|
|
|
|
|
|
|
|
400
|
1
|
50
|
|
|
|
380
|
if ($option{stdout}) { |
|
401
|
0
|
|
|
|
|
0
|
open $logfh, ">>&STDOUT"; |
|
402
|
|
|
|
|
|
|
} |
|
403
|
|
|
|
|
|
|
else { |
|
404
|
1
|
|
|
|
|
3
|
my $dopt = $option{logdate}; |
|
405
|
1
|
0
|
|
|
|
12
|
my $date = |
|
|
|
50
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
406
|
|
|
|
|
|
|
( "none" eq $dopt ) ? '' |
|
407
|
|
|
|
|
|
|
: ( "date" eq $dopt ) ? $starttime->ymd('') |
|
408
|
|
|
|
|
|
|
: ( "time" eq $dopt ) ? $starttime->hms('') |
|
409
|
|
|
|
|
|
|
|
|
410
|
|
|
|
|
|
|
# : ("datetime" eq $dopt) # validated, so must be 'datetime ' |
|
411
|
|
|
|
|
|
|
: ( $starttime->ymd('') . $starttime->hms('') ); |
|
412
|
1
|
50
|
|
|
|
16
|
$date .= '-' if $date; |
|
413
|
1
|
50
|
|
|
|
7
|
$option{logext} = ".$option{logext}" if $option{logext} =~ m(^[^.]); |
|
414
|
1
|
|
|
|
|
4
|
my $log_path = "$option{logdir}/$date$option{logname}$option{logext}"; |
|
415
|
1
|
50
|
|
|
|
137
|
open( $logfh, ">>", $log_path ) |
|
416
|
|
|
|
|
|
|
or carp "cannot open log file $log_path: $!"; |
|
417
|
1
|
|
|
|
|
157
|
say STDERR "Appending log info to $log_path"; |
|
418
|
1
|
|
|
|
|
4
|
my $lock_cnt = 0; |
|
419
|
1
|
|
|
|
|
1
|
while (1) { |
|
420
|
1
|
50
|
|
|
|
11
|
flock $logfh, LOCK_EX and last; |
|
421
|
0
|
0
|
|
|
|
0
|
croak "$0 [$$]: flock failed on $log_path: $!" if $lock_cnt > 30; |
|
422
|
0
|
0
|
|
|
|
0
|
say STDERR "Waiting for lock on $log_path" unless $lock_cnt++; |
|
423
|
0
|
|
|
|
|
0
|
print STDERR "."; |
|
424
|
0
|
|
|
|
|
0
|
sleep(2); |
|
425
|
|
|
|
|
|
|
} |
|
426
|
1
|
50
|
|
|
|
4
|
say "" if $lock_cnt; |
|
427
|
|
|
|
|
|
|
} |
|
428
|
|
|
|
|
|
|
|
|
429
|
1
|
|
|
|
|
24
|
say $logfh join( ' ', "########", $uid, '(', $gid, ') :', $progbase, @args ); |
|
430
|
|
|
|
|
|
|
|
|
431
|
1
|
|
|
|
|
5
|
my $mod = show_modules(); |
|
432
|
1
|
|
|
|
|
68
|
for my $key ( sort keys %$mod ) { |
|
433
|
109
|
|
|
|
|
80
|
my ( $ver, $loc ) = @{ $mod->{$key} }; |
|
|
109
|
|
|
|
|
152
|
|
|
434
|
109
|
|
|
|
|
113
|
log_entry( $logfh, MODULE => NAME => $key ); |
|
435
|
109
|
|
|
|
|
122
|
log_entry( $logfh, MODULE => VERSION => $ver ); |
|
436
|
109
|
|
|
|
|
120
|
log_entry( $logfh, MODULE => LOC => $loc ); |
|
437
|
|
|
|
|
|
|
} |
|
438
|
1
|
|
|
|
|
12
|
for my $inc (@INC) { |
|
439
|
11
|
|
|
|
|
13
|
log_entry( $logfh, INC => $inc ); |
|
440
|
|
|
|
|
|
|
} |
|
441
|
|
|
|
|
|
|
|
|
442
|
5
|
|
|
|
|
15388
|
log_entry( $logfh, UNAME => $_->[1], do { my $out = qx( uname $_->[0] ); chomp $out; $out } ) |
|
|
5
|
|
|
|
|
47
|
|
|
|
5
|
|
|
|
|
78
|
|
|
443
|
1
|
|
|
|
|
11
|
for ( |
|
444
|
|
|
|
|
|
|
[ -s => "System" ], |
|
445
|
|
|
|
|
|
|
[ -n => "Name" ], |
|
446
|
|
|
|
|
|
|
[ -r => "OSRel" ], |
|
447
|
|
|
|
|
|
|
[ -v => "OSVer" ], |
|
448
|
|
|
|
|
|
|
[ -m => "Machine" ] |
|
449
|
|
|
|
|
|
|
); |
|
450
|
1
|
|
|
|
|
19
|
log_entry( $logfh, PERL => $^X ); |
|
451
|
1
|
|
|
|
|
5
|
log_entry( $logfh, PERLVer => $] ); |
|
452
|
1
|
|
|
|
|
4
|
log_entry( $logfh, User => $uid ); |
|
453
|
1
|
|
|
|
|
4
|
log_entry( $logfh, Group => $gid ); |
|
454
|
1
|
|
|
|
|
6
|
log_entry( $logfh, ProgDir => $FindBin::Bin ); |
|
455
|
1
|
|
|
|
|
5
|
log_entry( $logfh, Program => $progbase ); |
|
456
|
1
|
|
50
|
|
|
16
|
log_entry( $logfh, Version => ( $::VERSION // "(No VERSION)" ) ); |
|
457
|
1
|
|
|
|
|
3
|
log_entry( $logfh, Args => scalar(@args) ); |
|
458
|
1
|
|
|
|
|
3
|
my $acnt = 0; |
|
459
|
1
|
|
|
|
|
5
|
log_entry( $logfh, " arg" => sprintf("%8d", ++$acnt), $args[$acnt-1] ) for @args; |
|
460
|
1
|
|
|
|
|
12
|
log_entry( $logfh, Start => $starttime->datetime() . "." . sprintf( "%03d", $starttime->millisecond ) ); |
|
461
|
1
|
|
|
|
|
6
|
log_entry( $logfh, End => $endtime->datetime() . "." . sprintf( "%03d", $endtime->millisecond ) ); |
|
462
|
1
|
|
|
|
|
12
|
my $dur = $endtime->subtract_datetime_absolute($starttime); |
|
463
|
1
|
|
|
|
|
242
|
log_entry( $logfh, Elapsed => $dur->delta_seconds . "." . |
|
464
|
|
|
|
|
|
|
sprintf( "%03d", $dur->delta_nanoseconds/1_000_000) ); |
|
465
|
1
|
|
|
|
|
5
|
log_entry( $logfh, EndStat => $exit_status ); |
|
466
|
|
|
|
|
|
|
|
|
467
|
1
|
|
|
0
|
|
3
|
$_->(sub { log_entry( $logfh, @_ ) }) for @extra_loggers; |
|
|
0
|
|
|
|
|
0
|
|
|
468
|
|
|
|
|
|
|
|
|
469
|
1
|
|
|
|
|
3
|
say $logfh ""; |
|
470
|
|
|
|
|
|
|
|
|
471
|
1
|
|
|
|
|
325
|
close($logfh); |
|
472
|
|
|
|
|
|
|
} |
|
473
|
|
|
|
|
|
|
} |
|
474
|
|
|
|
|
|
|
|
|
475
|
|
|
|
|
|
|
# Print version and loading path information for modules |
|
476
|
|
|
|
|
|
|
sub show_modules { |
|
477
|
1
|
|
|
1
|
0
|
1
|
my $module_infos = {}; |
|
478
|
|
|
|
|
|
|
|
|
479
|
|
|
|
|
|
|
# %INC looks like this: |
|
480
|
|
|
|
|
|
|
# { |
|
481
|
|
|
|
|
|
|
# ... |
|
482
|
|
|
|
|
|
|
# "Data/Dump.pm" |
|
483
|
|
|
|
|
|
|
# => "/whatever/perl/lib/site_perl/5.18.1/Data/Dump.pm", |
|
484
|
|
|
|
|
|
|
# ... |
|
485
|
|
|
|
|
|
|
# } |
|
486
|
|
|
|
|
|
|
# So let's convert it to this: |
|
487
|
|
|
|
|
|
|
# { |
|
488
|
|
|
|
|
|
|
# ... |
|
489
|
|
|
|
|
|
|
# "Data::Dump" |
|
490
|
|
|
|
|
|
|
# => [ "1.4.2", |
|
491
|
|
|
|
|
|
|
# "/whatever/perl/lib/site_perl/5.18.1/Data/Dump.pm", |
|
492
|
|
|
|
|
|
|
# ], |
|
493
|
|
|
|
|
|
|
# ... |
|
494
|
|
|
|
|
|
|
# } |
|
495
|
1
|
|
|
|
|
103
|
foreach my $module_inc_name ( keys(%INC) ) { |
|
496
|
109
|
|
|
|
|
101
|
my $real_name = $module_inc_name; |
|
497
|
109
|
|
|
|
|
186
|
$real_name =~ s|/|::|g; |
|
498
|
109
|
|
|
|
|
209
|
$real_name =~ s|\.pm$||; |
|
499
|
|
|
|
|
|
|
|
|
500
|
109
|
|
|
|
|
1278
|
my $version = eval { $real_name->VERSION } |
|
501
|
109
|
|
66
|
|
|
84
|
// eval { ${"${real_name}::VERSION"} } |
|
|
23
|
|
100
|
|
|
22
|
|
|
|
23
|
|
|
|
|
121
|
|
|
502
|
|
|
|
|
|
|
// 'unknown'; |
|
503
|
|
|
|
|
|
|
# stringify, in case it is a weird format |
|
504
|
|
|
|
|
|
|
# - I don't think the 'invalid' alternative can be hit, but safer to have it in |
|
505
|
109
|
|
50
|
|
|
138
|
$version = eval { $version . '' } // 'invalid'; |
|
|
109
|
|
|
|
|
174
|
|
|
506
|
|
|
|
|
|
|
|
|
507
|
109
|
|
|
|
|
333
|
$module_infos->{$real_name} = [ $version, $INC{$module_inc_name} ]; |
|
508
|
|
|
|
|
|
|
} |
|
509
|
|
|
|
|
|
|
|
|
510
|
1
|
|
|
|
|
17
|
return $module_infos; |
|
511
|
|
|
|
|
|
|
} |
|
512
|
|
|
|
|
|
|
|
|
513
|
|
|
|
|
|
|
1; |