| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Parse::Syslog; |
|
2
|
|
|
|
|
|
|
|
|
3
|
8
|
|
|
8
|
|
105163
|
use Carp; |
|
|
8
|
|
|
|
|
23
|
|
|
|
8
|
|
|
|
|
782
|
|
|
4
|
8
|
|
|
8
|
|
10912
|
use Symbol; |
|
|
8
|
|
|
|
|
9333
|
|
|
|
8
|
|
|
|
|
1022
|
|
|
5
|
8
|
|
|
8
|
|
10240
|
use Time::Local; |
|
|
8
|
|
|
|
|
24643
|
|
|
|
8
|
|
|
|
|
566
|
|
|
6
|
8
|
|
|
8
|
|
9076
|
use IO::File; |
|
|
8
|
|
|
|
|
91816
|
|
|
|
8
|
|
|
|
|
1165
|
|
|
7
|
8
|
|
|
8
|
|
75
|
use strict; |
|
|
8
|
|
|
|
|
17
|
|
|
|
8
|
|
|
|
|
292
|
|
|
8
|
8
|
|
|
8
|
|
41
|
use vars qw($VERSION); |
|
|
8
|
|
|
|
|
12
|
|
|
|
8
|
|
|
|
|
5664
|
|
|
9
|
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
$VERSION = '1.10'; |
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
my %months_map = ( |
|
13
|
|
|
|
|
|
|
'Jan' => 0, 'Feb' => 1, 'Mar' => 2, |
|
14
|
|
|
|
|
|
|
'Apr' => 3, 'May' => 4, 'Jun' => 5, |
|
15
|
|
|
|
|
|
|
'Jul' => 6, 'Aug' => 7, 'Sep' => 8, |
|
16
|
|
|
|
|
|
|
'Oct' => 9, 'Nov' =>10, 'Dec' =>11, |
|
17
|
|
|
|
|
|
|
'jan' => 0, 'feb' => 1, 'mar' => 2, |
|
18
|
|
|
|
|
|
|
'apr' => 3, 'may' => 4, 'jun' => 5, |
|
19
|
|
|
|
|
|
|
'jul' => 6, 'aug' => 7, 'sep' => 8, |
|
20
|
|
|
|
|
|
|
'oct' => 9, 'nov' =>10, 'dec' =>11, |
|
21
|
|
|
|
|
|
|
); |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
sub is_dst_switch($$$) |
|
24
|
|
|
|
|
|
|
{ |
|
25
|
0
|
|
|
0
|
0
|
0
|
my ($self, $t, $time) = @_; |
|
26
|
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
# calculate the time in one hour and see if the difference is 3600 seconds. |
|
28
|
|
|
|
|
|
|
# if not, we are in a dst-switch hour |
|
29
|
|
|
|
|
|
|
# note that right now we only support 1-hour dst offsets |
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
# cache the result |
|
32
|
0
|
0
|
0
|
|
|
0
|
if(defined $self->{is_dst_switch_last_hour} and |
|
33
|
|
|
|
|
|
|
$self->{is_dst_switch_last_hour} == $t->[3]<<5+$t->[2]) { |
|
34
|
0
|
|
|
|
|
0
|
return @{$self->{is_dst_switch_result}}; |
|
|
0
|
|
|
|
|
0
|
|
|
35
|
|
|
|
|
|
|
} |
|
36
|
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
# calculate a number out of the day and hour to identify the hour |
|
38
|
0
|
|
|
|
|
0
|
$self->{is_dst_switch_last_hour} = $t->[3]<<5+$t->[2]; |
|
39
|
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
# calculating hour+1 (below) is a problem if the hour is 23. as far as I |
|
41
|
|
|
|
|
|
|
# know, nobody does the DST switch at this time, so just assume it isn't |
|
42
|
|
|
|
|
|
|
# DST switch if the hour is 23. |
|
43
|
0
|
0
|
|
|
|
0
|
if($t->[2]==23) { |
|
44
|
0
|
|
|
|
|
0
|
@{$self->{is_dst_switch_result}} = (0, undef); |
|
|
0
|
|
|
|
|
0
|
|
|
45
|
0
|
|
|
|
|
0
|
return @{$self->{is_dst_switch_result}}; |
|
|
0
|
|
|
|
|
0
|
|
|
46
|
|
|
|
|
|
|
} |
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
# let's see the timestamp in one hour |
|
49
|
|
|
|
|
|
|
# 0: sec, 1: min, 2: h, 3: day, 4: month, 5: year |
|
50
|
0
|
|
|
|
|
0
|
my $time_plus_1h = timelocal($t->[0], $t->[1], $t->[2]+1, $t->[3], $t->[4], $t->[5]); |
|
51
|
|
|
|
|
|
|
|
|
52
|
0
|
0
|
|
|
|
0
|
if($time_plus_1h - $time > 4000) { |
|
53
|
0
|
|
|
|
|
0
|
@{$self->{is_dst_switch_result}} = (3600, $time-$time%3600+3600); |
|
|
0
|
|
|
|
|
0
|
|
|
54
|
|
|
|
|
|
|
} |
|
55
|
|
|
|
|
|
|
else { |
|
56
|
0
|
|
|
|
|
0
|
@{$self->{is_dst_switch_result}} = (0, undef); |
|
|
0
|
|
|
|
|
0
|
|
|
57
|
|
|
|
|
|
|
} |
|
58
|
|
|
|
|
|
|
|
|
59
|
0
|
|
|
|
|
0
|
return @{$self->{is_dst_switch_result}}; |
|
|
0
|
|
|
|
|
0
|
|
|
60
|
|
|
|
|
|
|
} |
|
61
|
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
# fast timelocal, cache minute's timestamp |
|
63
|
|
|
|
|
|
|
# don't cache more than minute because of daylight saving time switch |
|
64
|
|
|
|
|
|
|
# 0: sec, 1: min, 2: h, 3: day, 4: month, 5: year |
|
65
|
|
|
|
|
|
|
sub str2time($$$$$$$$) |
|
66
|
|
|
|
|
|
|
{ |
|
67
|
62
|
|
|
62
|
0
|
88
|
my $self = shift @_; |
|
68
|
62
|
|
|
|
|
115
|
my $GMT = pop @_; |
|
69
|
|
|
|
|
|
|
|
|
70
|
62
|
|
|
|
|
227
|
my $lastmin = $self->{str2time_lastmin}; |
|
71
|
62
|
100
|
100
|
|
|
720
|
if(defined $lastmin and |
|
|
|
|
66
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
72
|
|
|
|
|
|
|
$lastmin->[0] == $_[1] and |
|
73
|
|
|
|
|
|
|
$lastmin->[1] == $_[2] and |
|
74
|
|
|
|
|
|
|
$lastmin->[2] == $_[3] and |
|
75
|
|
|
|
|
|
|
$lastmin->[3] == $_[4] and |
|
76
|
|
|
|
|
|
|
$lastmin->[4] == $_[5]) |
|
77
|
|
|
|
|
|
|
{ |
|
78
|
22
|
|
|
|
|
58
|
$self->{last_time} = $self->{str2time_lastmin_time} + $_[0]; |
|
79
|
22
|
|
50
|
|
|
124
|
return $self->{last_time} + ($self->{dst_comp}||0); |
|
80
|
|
|
|
|
|
|
} |
|
81
|
|
|
|
|
|
|
|
|
82
|
40
|
|
|
|
|
48
|
my $time; |
|
83
|
40
|
100
|
|
|
|
64
|
if($GMT) { |
|
84
|
6
|
|
|
|
|
27
|
$time = timegm(@_); |
|
85
|
|
|
|
|
|
|
} |
|
86
|
|
|
|
|
|
|
else { |
|
87
|
34
|
|
|
|
|
118
|
$time = timelocal(@_); |
|
88
|
|
|
|
|
|
|
} |
|
89
|
|
|
|
|
|
|
|
|
90
|
|
|
|
|
|
|
# compensate for DST-switch |
|
91
|
|
|
|
|
|
|
# - if a timewarp is detected (1:00 -> 1:30 -> 1:00): |
|
92
|
|
|
|
|
|
|
# - test if we are in a DST-switch-hour |
|
93
|
|
|
|
|
|
|
# - compensate if yes |
|
94
|
|
|
|
|
|
|
# note that we assume that the DST-switch goes like this: |
|
95
|
|
|
|
|
|
|
# time 1:00 1:30 2:00 2:30 2:00 2:30 3:00 3:30 |
|
96
|
|
|
|
|
|
|
# stamp 1 2 3 4 3 3 7 8 |
|
97
|
|
|
|
|
|
|
# comp. 0 0 0 0 2 2 0 0 |
|
98
|
|
|
|
|
|
|
# result 1 2 3 4 5 6 7 8 |
|
99
|
|
|
|
|
|
|
# old Time::Local versions behave differently (1 2 5 6 5 6 7 8) |
|
100
|
|
|
|
|
|
|
|
|
101
|
40
|
50
|
66
|
|
|
2749
|
if(!$GMT and !defined $self->{dst_comp} and |
|
|
|
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
102
|
|
|
|
|
|
|
defined $self->{last_time} and |
|
103
|
|
|
|
|
|
|
$self->{last_time}-$time > 1200 and |
|
104
|
|
|
|
|
|
|
$self->{last_time}-$time < 3600) |
|
105
|
|
|
|
|
|
|
{ |
|
106
|
0
|
|
|
|
|
0
|
my ($off, $until) = $self->is_dst_switch(\@_, $time); |
|
107
|
0
|
0
|
|
|
|
0
|
if($off) { |
|
108
|
0
|
|
|
|
|
0
|
$self->{dst_comp} = $off; |
|
109
|
0
|
|
|
|
|
0
|
$self->{dst_comp_until} = $until; |
|
110
|
|
|
|
|
|
|
} |
|
111
|
|
|
|
|
|
|
} |
|
112
|
40
|
50
|
33
|
|
|
143
|
if(defined $self->{dst_comp_until} and $time > $self->{dst_comp_until}) { |
|
113
|
0
|
|
|
|
|
0
|
delete $self->{dst_comp}; |
|
114
|
0
|
|
|
|
|
0
|
delete $self->{dst_comp_until}; |
|
115
|
|
|
|
|
|
|
} |
|
116
|
|
|
|
|
|
|
|
|
117
|
40
|
|
|
|
|
196
|
$self->{str2time_lastmin} = [ @_[1..5] ]; |
|
118
|
40
|
|
|
|
|
100
|
$self->{str2time_lastmin_time} = $time-$_[0]; |
|
119
|
40
|
|
|
|
|
56
|
$self->{last_time} = $time; |
|
120
|
40
|
|
50
|
|
|
219
|
return $time+($self->{dst_comp}||0); |
|
121
|
|
|
|
|
|
|
} |
|
122
|
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
sub _use_locale($) |
|
124
|
|
|
|
|
|
|
{ |
|
125
|
8
|
|
|
8
|
|
8239
|
use POSIX qw(locale_h strftime); |
|
|
8
|
|
|
|
|
75457
|
|
|
|
8
|
|
|
|
|
58
|
|
|
126
|
0
|
|
|
0
|
|
0
|
my $old_locale = setlocale(LC_TIME); |
|
127
|
0
|
|
|
|
|
0
|
for my $locale (@_) { |
|
128
|
0
|
0
|
|
|
|
0
|
croak "new(): wrong 'locale' value: '$locale'" unless setlocale(LC_TIME, $locale); |
|
129
|
0
|
|
|
|
|
0
|
for my $month (0..11) { |
|
130
|
0
|
|
|
|
|
0
|
$months_map{strftime("%b", 0, 0, 0, 1, $month, 96)} = $month; |
|
131
|
|
|
|
|
|
|
} |
|
132
|
|
|
|
|
|
|
} |
|
133
|
0
|
|
|
|
|
0
|
setlocale(LC_TIME, $old_locale); |
|
134
|
|
|
|
|
|
|
} |
|
135
|
|
|
|
|
|
|
|
|
136
|
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
sub new($$;%) |
|
138
|
|
|
|
|
|
|
{ |
|
139
|
7
|
|
|
7
|
0
|
3970
|
my ($class, $file, %data) = @_; |
|
140
|
7
|
50
|
|
|
|
112
|
croak "new() requires one argument: file" unless defined $file; |
|
141
|
7
|
50
|
|
|
|
28
|
%data = () unless %data; |
|
142
|
7
|
50
|
|
|
|
49
|
if(not defined $data{year}) { |
|
143
|
0
|
|
|
|
|
0
|
$data{year} = (localtime(time))[5]+1900; |
|
144
|
|
|
|
|
|
|
} |
|
145
|
7
|
100
|
|
|
|
43
|
$data{type} = 'syslog' unless defined $data{type}; |
|
146
|
7
|
|
|
|
|
20
|
$data{_repeat}=0; |
|
147
|
|
|
|
|
|
|
|
|
148
|
7
|
50
|
|
|
|
119
|
if(UNIVERSAL::isa($file, 'IO::Handle')) { |
|
|
|
50
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
149
|
0
|
|
|
|
|
0
|
$data{file} = $file; |
|
150
|
|
|
|
|
|
|
} |
|
151
|
|
|
|
|
|
|
elsif(UNIVERSAL::isa($file, 'File::Tail')) { |
|
152
|
0
|
|
|
|
|
0
|
$data{file} = $file; |
|
153
|
0
|
|
|
|
|
0
|
$data{filetail}=1; |
|
154
|
|
|
|
|
|
|
} |
|
155
|
|
|
|
|
|
|
elsif(! ref $file) { |
|
156
|
7
|
50
|
|
|
|
26
|
if($file eq '-') { |
|
157
|
0
|
|
|
|
|
0
|
my $io = new IO::Handle; |
|
158
|
0
|
|
|
|
|
0
|
$data{file} = $io->fdopen(fileno(STDIN),"r"); |
|
159
|
|
|
|
|
|
|
} |
|
160
|
|
|
|
|
|
|
else { |
|
161
|
7
|
|
|
|
|
69
|
$data{file} = new IO::File($file, "<"); |
|
162
|
7
|
50
|
|
|
|
903
|
defined $data{file} or croak "can't open $file: $!"; |
|
163
|
|
|
|
|
|
|
} |
|
164
|
|
|
|
|
|
|
} |
|
165
|
|
|
|
|
|
|
else { |
|
166
|
0
|
|
|
|
|
0
|
croak "argument must be either a file-name or an IO::Handle object."; |
|
167
|
|
|
|
|
|
|
} |
|
168
|
|
|
|
|
|
|
|
|
169
|
7
|
50
|
|
|
|
31
|
if(defined $data{locale}) { |
|
170
|
0
|
0
|
|
|
|
0
|
if(ref $data{locale} eq 'ARRAY') { |
|
|
|
0
|
|
|
|
|
|
|
171
|
0
|
|
|
|
|
0
|
_use_locale @{$data{locale}}; |
|
|
0
|
|
|
|
|
0
|
|
|
172
|
|
|
|
|
|
|
} |
|
173
|
|
|
|
|
|
|
elsif(ref $data{locale} eq '') { |
|
174
|
0
|
|
|
|
|
0
|
_use_locale $data{locale}; |
|
175
|
|
|
|
|
|
|
} |
|
176
|
|
|
|
|
|
|
else { |
|
177
|
0
|
|
|
|
|
0
|
croak "'locale' parameter must be scalar or array of scalars"; |
|
178
|
|
|
|
|
|
|
} |
|
179
|
|
|
|
|
|
|
} |
|
180
|
|
|
|
|
|
|
|
|
181
|
7
|
|
|
|
|
31
|
return bless \%data, $class; |
|
182
|
|
|
|
|
|
|
} |
|
183
|
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
sub _year_increment($$) |
|
185
|
|
|
|
|
|
|
{ |
|
186
|
62
|
|
|
62
|
|
88
|
my ($self, $mon) = @_; |
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
# year change |
|
189
|
62
|
100
|
|
|
|
160
|
if($mon==0) { |
|
|
|
100
|
|
|
|
|
|
|
190
|
7
|
100
|
66
|
|
|
62
|
$self->{year}++ if defined $self->{_last_mon} and $self->{_last_mon} == 11; |
|
191
|
7
|
|
|
|
|
12
|
$self->{enable_year_decrement} = 1; |
|
192
|
|
|
|
|
|
|
} |
|
193
|
|
|
|
|
|
|
elsif($mon == 11) { |
|
194
|
3
|
100
|
|
|
|
10
|
if($self->{enable_year_decrement}) { |
|
195
|
1
|
50
|
33
|
|
|
9
|
$self->{year}-- if defined $self->{_last_mon} and $self->{_last_mon} != 11; |
|
196
|
|
|
|
|
|
|
} |
|
197
|
|
|
|
|
|
|
} |
|
198
|
|
|
|
|
|
|
else { |
|
199
|
52
|
|
|
|
|
119
|
$self->{enable_year_decrement} = 0; |
|
200
|
|
|
|
|
|
|
} |
|
201
|
|
|
|
|
|
|
|
|
202
|
62
|
|
|
|
|
124
|
$self->{_last_mon} = $mon; |
|
203
|
|
|
|
|
|
|
} |
|
204
|
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
sub _next_line($) |
|
206
|
|
|
|
|
|
|
{ |
|
207
|
69
|
|
|
69
|
|
82
|
my $self = shift; |
|
208
|
69
|
|
|
|
|
104
|
my $f = $self->{file}; |
|
209
|
69
|
50
|
|
|
|
142
|
if(defined $self->{filetail}) { |
|
210
|
0
|
|
|
|
|
0
|
return $f->read; |
|
211
|
|
|
|
|
|
|
} |
|
212
|
|
|
|
|
|
|
else { |
|
213
|
69
|
|
|
|
|
2000
|
return $f->getline; |
|
214
|
|
|
|
|
|
|
} |
|
215
|
|
|
|
|
|
|
} |
|
216
|
|
|
|
|
|
|
|
|
217
|
|
|
|
|
|
|
sub _next_syslog($) |
|
218
|
|
|
|
|
|
|
{ |
|
219
|
146
|
|
|
146
|
|
174
|
my ($self) = @_; |
|
220
|
|
|
|
|
|
|
|
|
221
|
146
|
|
|
|
|
359
|
while($self->{_repeat}>0) { |
|
222
|
84
|
|
|
|
|
104
|
$self->{_repeat}--; |
|
223
|
84
|
|
|
|
|
229
|
return $self->{_repeat_data}; |
|
224
|
|
|
|
|
|
|
} |
|
225
|
|
|
|
|
|
|
|
|
226
|
62
|
|
|
|
|
97
|
my $file = $self->{file}; |
|
227
|
62
|
|
|
|
|
138
|
line: while(defined (my $str = $self->_next_line)) { |
|
228
|
|
|
|
|
|
|
# date, time and host |
|
229
|
|
|
|
|
|
|
$str =~ /^ |
|
230
|
|
|
|
|
|
|
(\S{3})\s+(\d+) # date -- 1, 2 |
|
231
|
|
|
|
|
|
|
\s |
|
232
|
|
|
|
|
|
|
(\d+):(\d+):(\d+) # time -- 3, 4, 5 |
|
233
|
|
|
|
|
|
|
(?:\s<\w+\.\w+>)? # FreeBSD's verbose-mode |
|
234
|
|
|
|
|
|
|
\s |
|
235
|
|
|
|
|
|
|
([-\w\.\@:]+) # host -- 6 |
|
236
|
|
|
|
|
|
|
\s+ |
|
237
|
|
|
|
|
|
|
(?:\[LOG_[A-Z]+\]\s+)? # FreeBSD |
|
238
|
|
|
|
|
|
|
(.*) # text -- 7 |
|
239
|
|
|
|
|
|
|
$/x or do |
|
240
|
59
|
50
|
|
|
|
2181
|
{ |
|
241
|
0
|
|
|
|
|
0
|
warn "WARNING: line not in syslog format: $str"; |
|
242
|
0
|
|
|
|
|
0
|
next line; |
|
243
|
|
|
|
|
|
|
}; |
|
244
|
|
|
|
|
|
|
|
|
245
|
59
|
|
|
|
|
127
|
my $mon = $months_map{$1}; |
|
246
|
59
|
50
|
|
|
|
120
|
defined $mon or croak "unknown month $1\n"; |
|
247
|
|
|
|
|
|
|
|
|
248
|
59
|
|
|
|
|
205
|
$self->_year_increment($mon); |
|
249
|
|
|
|
|
|
|
|
|
250
|
|
|
|
|
|
|
# convert to unix time |
|
251
|
59
|
|
|
|
|
280
|
my $time = $self->str2time($5,$4,$3,$2,$mon,$self->{year}-1900,$self->{GMT}); |
|
252
|
59
|
50
|
|
|
|
173
|
if(not $self->{allow_future}) { |
|
253
|
|
|
|
|
|
|
# accept maximum one day in the present future |
|
254
|
59
|
50
|
|
|
|
186
|
if($time - time > 86400) { |
|
255
|
0
|
|
|
|
|
0
|
warn "WARNING: ignoring future date in syslog line: $str"; |
|
256
|
0
|
|
|
|
|
0
|
next line; |
|
257
|
|
|
|
|
|
|
} |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
|
|
260
|
59
|
|
|
|
|
160
|
my ($host, $text) = ($6, $7); |
|
261
|
|
|
|
|
|
|
|
|
262
|
|
|
|
|
|
|
# last message repeated ... times |
|
263
|
59
|
100
|
|
|
|
224
|
if($text =~ /^(?:last message repeated|above message repeats) (\d+) time/) { |
|
264
|
14
|
50
|
33
|
|
|
58
|
next line if defined $self->{repeat} and not $self->{repeat}; |
|
265
|
14
|
100
|
|
|
|
64
|
next line if not defined $self->{_last_data}{$host}; |
|
266
|
12
|
50
|
|
|
|
40
|
$1 > 0 or do { |
|
267
|
0
|
|
|
|
|
0
|
warn "WARNING: last message repeated 0 or less times??\n"; |
|
268
|
0
|
|
|
|
|
0
|
next line; |
|
269
|
|
|
|
|
|
|
}; |
|
270
|
12
|
|
|
|
|
27
|
$self->{_repeat}=$1-1; |
|
271
|
12
|
|
|
|
|
32
|
$self->{_repeat_data}=$self->{_last_data}{$host}; |
|
272
|
12
|
|
|
|
|
54
|
return $self->{_last_data}{$host}; |
|
273
|
|
|
|
|
|
|
} |
|
274
|
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
# marks |
|
276
|
45
|
100
|
|
|
|
99
|
next if $text eq '-- MARK --'; |
|
277
|
|
|
|
|
|
|
|
|
278
|
|
|
|
|
|
|
# some systems send over the network their |
|
279
|
|
|
|
|
|
|
# hostname prefixed to the text. strip that. |
|
280
|
44
|
|
|
|
|
479
|
$text =~ s/^$host\s+//; |
|
281
|
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
# discard ':' in HP-UX 'su' entries like this: |
|
283
|
|
|
|
|
|
|
# Apr 24 19:09:40 remedy : su : + tty?? root-oracle |
|
284
|
44
|
|
|
|
|
77
|
$text =~ s/^:\s+//; |
|
285
|
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
$text =~ /^ |
|
287
|
|
|
|
|
|
|
([^:]+?) # program -- 1 |
|
288
|
|
|
|
|
|
|
(?:\[(\d+)\])? # PID -- 2 |
|
289
|
|
|
|
|
|
|
:\s+ |
|
290
|
|
|
|
|
|
|
(?:\[ID\ (\d+)\ ([a-z0-9]+)\.([a-z]+)\]\ )? # Solaris 8 "message id" -- 3, 4, 5 |
|
291
|
|
|
|
|
|
|
(.*) # text -- 6 |
|
292
|
|
|
|
|
|
|
$/x or do |
|
293
|
44
|
50
|
|
|
|
340
|
{ |
|
294
|
0
|
|
|
|
|
0
|
warn "WARNING: line not in syslog format: $str"; |
|
295
|
0
|
|
|
|
|
0
|
next line; |
|
296
|
|
|
|
|
|
|
}; |
|
297
|
|
|
|
|
|
|
|
|
298
|
44
|
50
|
|
|
|
129
|
if($self->{arrayref}) { |
|
299
|
0
|
|
|
|
|
0
|
$self->{_last_data}{$host} = [ |
|
300
|
|
|
|
|
|
|
$time, # 0: timestamp |
|
301
|
|
|
|
|
|
|
$host, # 1: host |
|
302
|
|
|
|
|
|
|
$1, # 2: program |
|
303
|
|
|
|
|
|
|
$2, # 3: pid |
|
304
|
|
|
|
|
|
|
$6, # 4: text |
|
305
|
|
|
|
|
|
|
]; |
|
306
|
|
|
|
|
|
|
} |
|
307
|
|
|
|
|
|
|
else { |
|
308
|
44
|
|
|
|
|
495
|
$self->{_last_data}{$host} = { |
|
309
|
|
|
|
|
|
|
timestamp => $time, |
|
310
|
|
|
|
|
|
|
host => $host, |
|
311
|
|
|
|
|
|
|
program => $1, |
|
312
|
|
|
|
|
|
|
pid => $2, |
|
313
|
|
|
|
|
|
|
msgid => $3, |
|
314
|
|
|
|
|
|
|
facility => $4, |
|
315
|
|
|
|
|
|
|
level => $5, |
|
316
|
|
|
|
|
|
|
text => $6, |
|
317
|
|
|
|
|
|
|
}; |
|
318
|
|
|
|
|
|
|
} |
|
319
|
|
|
|
|
|
|
|
|
320
|
44
|
|
|
|
|
246
|
return $self->{_last_data}{$host}; |
|
321
|
|
|
|
|
|
|
} |
|
322
|
6
|
|
|
|
|
236
|
return undef; |
|
323
|
|
|
|
|
|
|
} |
|
324
|
|
|
|
|
|
|
|
|
325
|
|
|
|
|
|
|
sub _next_metalog($) |
|
326
|
|
|
|
|
|
|
{ |
|
327
|
4
|
|
|
4
|
|
5
|
my ($self) = @_; |
|
328
|
4
|
|
|
|
|
6
|
my $file = $self->{file}; |
|
329
|
4
|
|
|
|
|
10
|
line: while(my $str = $self->_next_line) { |
|
330
|
|
|
|
|
|
|
# date, time and host |
|
331
|
|
|
|
|
|
|
|
|
332
|
|
|
|
|
|
|
$str =~ /^ |
|
333
|
|
|
|
|
|
|
(\S{3})\s+(\d+) # date -- 1, 2 |
|
334
|
|
|
|
|
|
|
\s |
|
335
|
|
|
|
|
|
|
(\d+):(\d+):(\d+) # time -- 3, 4, 5 |
|
336
|
|
|
|
|
|
|
# host is not logged |
|
337
|
|
|
|
|
|
|
\s+ |
|
338
|
|
|
|
|
|
|
(.*) # text -- 6 |
|
339
|
|
|
|
|
|
|
$/x or do |
|
340
|
3
|
50
|
|
|
|
119
|
{ |
|
341
|
0
|
|
|
|
|
0
|
warn "WARNING: line not in metalog format: $str"; |
|
342
|
0
|
|
|
|
|
0
|
next line; |
|
343
|
|
|
|
|
|
|
}; |
|
344
|
|
|
|
|
|
|
|
|
345
|
3
|
|
|
|
|
7
|
my $mon = $months_map{$1}; |
|
346
|
3
|
50
|
|
|
|
9
|
defined $mon or croak "unknown month $1\n"; |
|
347
|
|
|
|
|
|
|
|
|
348
|
3
|
|
|
|
|
7
|
$self->_year_increment($mon); |
|
349
|
|
|
|
|
|
|
|
|
350
|
|
|
|
|
|
|
# convert to unix time |
|
351
|
3
|
|
|
|
|
17
|
my $time = $self->str2time($5,$4,$3,$2,$mon,$self->{year}-1900,$self->{GMT}); |
|
352
|
|
|
|
|
|
|
|
|
353
|
3
|
|
|
|
|
8
|
my $text = $6; |
|
354
|
|
|
|
|
|
|
|
|
355
|
|
|
|
|
|
|
$text =~ /^ |
|
356
|
|
|
|
|
|
|
\[(.*?)\] # program -- 1 |
|
357
|
|
|
|
|
|
|
# no PID |
|
358
|
|
|
|
|
|
|
\s+ |
|
359
|
|
|
|
|
|
|
(.*) # text -- 2 |
|
360
|
|
|
|
|
|
|
$/x or do |
|
361
|
3
|
50
|
|
|
|
15
|
{ |
|
362
|
0
|
|
|
|
|
0
|
warn "WARNING: text line not in metalog format: $text ($str)"; |
|
363
|
0
|
|
|
|
|
0
|
next line; |
|
364
|
|
|
|
|
|
|
}; |
|
365
|
|
|
|
|
|
|
|
|
366
|
3
|
50
|
|
|
|
5
|
if($self->{arrayref}) { |
|
367
|
|
|
|
|
|
|
return [ |
|
368
|
0
|
|
|
|
|
0
|
$time, # 0: timestamp |
|
369
|
|
|
|
|
|
|
'localhost', # 1: host |
|
370
|
|
|
|
|
|
|
$1, # 2: program |
|
371
|
|
|
|
|
|
|
undef, # 3: (no) pid |
|
372
|
|
|
|
|
|
|
$2, # 4: text |
|
373
|
|
|
|
|
|
|
]; |
|
374
|
|
|
|
|
|
|
} |
|
375
|
|
|
|
|
|
|
else { |
|
376
|
|
|
|
|
|
|
return { |
|
377
|
3
|
|
|
|
|
23
|
timestamp => $time, |
|
378
|
|
|
|
|
|
|
host => 'localhost', |
|
379
|
|
|
|
|
|
|
program => $1, |
|
380
|
|
|
|
|
|
|
text => $2, |
|
381
|
|
|
|
|
|
|
}; |
|
382
|
|
|
|
|
|
|
} |
|
383
|
|
|
|
|
|
|
} |
|
384
|
1
|
|
|
|
|
35
|
return undef; |
|
385
|
|
|
|
|
|
|
} |
|
386
|
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
sub next($) |
|
388
|
|
|
|
|
|
|
{ |
|
389
|
150
|
|
|
150
|
0
|
47254
|
my ($self) = @_; |
|
390
|
150
|
100
|
|
|
|
396
|
if($self->{type} eq 'syslog') { |
|
|
|
50
|
|
|
|
|
|
|
391
|
146
|
|
|
|
|
290
|
return $self->_next_syslog(); |
|
392
|
|
|
|
|
|
|
} |
|
393
|
|
|
|
|
|
|
elsif($self->{type} eq 'metalog') { |
|
394
|
4
|
|
|
|
|
9
|
return $self->_next_metalog(); |
|
395
|
|
|
|
|
|
|
} |
|
396
|
0
|
|
|
|
|
|
croak "Internal error: unknown type: $self->{type}"; |
|
397
|
|
|
|
|
|
|
} |
|
398
|
|
|
|
|
|
|
|
|
399
|
|
|
|
|
|
|
1; |
|
400
|
|
|
|
|
|
|
|
|
401
|
|
|
|
|
|
|
__END__ |