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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
|
#!/usr/bin/perl
#
# annotate-lintian-hints -- transform lintian tags into descriptive text
#
# Copyright (C) 1998 Christian Schwarz and Richard Braakman
# Copyright (C) 2013 Niels Thykier
# Copyright (C) 2017 Chris Lamb <lamby@debian.org>
# Copyright (C) 2020 Felix Lechner
#
# This program is free software. It is distributed under the terms of
# the GNU General Public License as published by the Free Software
# Foundation; either version 2 of the License, or (at your option) any
# later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, you can find it on the World Wide
# Web at https://www.gnu.org/copyleft/gpl.html, or write to the Free
# Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston,
# MA 02110-1301, USA.
use v5.20;
use warnings;
use utf8;
use Cwd qw(realpath);
use File::Basename qw(dirname);
# neither Path::This nor lib::relative are in Debian
use constant THISFILE => realpath __FILE__;
use constant THISDIR => dirname realpath __FILE__;
# use Lintian modules that belong to this program
use lib THISDIR . '/../lib';
# substituted during package build
my $LINTIAN_VERSION;
use Const::Fast;
use Getopt::Long ();
use IO::Interactive qw(is_interactive);
use Term::ReadKey;
use Unicode::UTF8 qw(encode_utf8 decode_utf8);
use Lintian::Output::EWI;
use Lintian::Profile;
use Lintian::Version qw(guess_version);
const my $EMPTY => q{};
const my $SPACE => q{ };
const my $DEFAULT_OUTPUT_WIDTH => 80;
const my $NEW_PROGRAM_NAME => q{lintian-annotate-hints};
my $TERMINAL_WIDTH;
($TERMINAL_WIDTH, undef, undef, undef) = GetTerminalSize()
if is_interactive;
$TERMINAL_WIDTH //= $DEFAULT_OUTPUT_WIDTH;
if (my $coverage_arg = $ENV{'LINTIAN_COVERAGE'}) {
my $p5opt = $ENV{'PERL5OPT'}//$EMPTY;
$p5opt .= $SPACE if $p5opt ne $EMPTY;
$ENV{'PERL5OPT'} = "${p5opt} ${coverage_arg}";
}
$ENV{LINTIAN_BASE} = realpath(THISDIR . '/..')
// die encode_utf8('Cannot resolve LINTIAN_BASE');
$ENV{LINTIAN_VERSION} = $LINTIAN_VERSION // guess_version($ENV{LINTIAN_BASE});
die encode_utf8('Unable to determine the version automatically!?')
unless length $ENV{LINTIAN_VERSION};
my $annotate;
my @INCLUDE_DIRS;
my $profile_name;
my $user_dirs = 1;
my %options = (
'annotate|a' => \$annotate,
'help|h' => \&show_help,
'include-dir=s' => \@INCLUDE_DIRS,
'output-width=i' => \$TERMINAL_WIDTH,
'profile=s' => \$profile_name,
'user-dirs!' => \$user_dirs,
'version' => \&show_version,
);
Getopt::Long::Configure('gnu_getopt');
Getopt::Long::GetOptions(%options)
or die encode_utf8("error parsing options\n");
my $profile = Lintian::Profile->new;
$profile->load($profile_name, \@INCLUDE_DIRS, $user_dirs);
my $output = Lintian::Output::EWI->new;
# Matches something like: (1:2.0-3) [arch1 arch2]
# - captures the version and the architectures
my $verarchre = qr{(?: \s* \(( [^)]++ )\) \s* \[ ( [^]]++ ) \])}x;
my $type_re = qr/(?:binary|changes|source|udeb)/;
my %already_displayed;
# Otherwise, read input files or STDIN, watch for tags, and add
# descriptions whenever we see one, can, and haven't already
# explained that tag.
while(my $bytes = <STDIN>) {
my $line = decode_utf8($bytes);
chomp $line;
say encode_utf8('N:');
say encode_utf8($line);
next
if $line =~ /^\s*$/;
# strip color
$line =~ s/\e[\[\d;]*m//g;
# strip HTML
$line =~ s/<span style=\"[^\"]+\">//g;
$line =~ s{</span>}{}g;
my $tag_name;
if ($annotate) {
# used for override files only; combine if possible
next
unless $line =~ m{^(?: # start optional part
(?:\S+)? # Optionally starts with package name
(?: \s*+ \[[^\]]+?\])? # optionally followed by an [arch-list] (like in B-D)
(?: \s*+ $type_re)? # optionally followed by the type
:\s++)? # end optional part
([\-\.a-zA-Z_0-9]+ (?:\s.+)?)$}x; # <tag-name> [extra] -> $1
my $tagdata = $1;
($tag_name, undef) = split(/ /, $tagdata, 2);
} elsif ($line
=~ m{^([^N]): (\S+)(?: (\S+)(?:$verarchre)?)?: (\S+)(?:\s+(.*))?$}) {
# matches the full deal:
# 1 222 3333 4444444 5555 666 777
# - T: pkg type (version) [arch]: tag [...]
# ^^^^^^^^^^^^^^^^^^^^^
# Where the marked part(s) are optional values. The numbers above
# the example are the capture groups.
my $pkg_type = $3 // 'binary';
$tag_name = $6;
} else {
next;
}
next
if $already_displayed{$tag_name}++;
my $tag = $profile->get_tag($tag_name);
next
unless defined $tag;
$output->describe_tag($profile->data, $tag, $TERMINAL_WIDTH);
}
exit;
sub show_version {
say encode_utf8("$NEW_PROGRAM_NAME v$ENV{LINTIAN_VERSION}");
exit;
}
sub show_help {
my $message =<<"EOT";
Usage: $NEW_PROGRAM_NAME [log-file...] ...
$NEW_PROGRAM_NAME --annotate [overrides ...]
Options:
-a, --annotate display descriptions of tags in Lintian overrides
--include-dir DIR check for Lintian data in DIR
--profile X use vendor profile X to determine severities
--output-width NUM set output width instead of probing terminal
--[no-]user-dirs whether to include profiles from user directories
--version show version info and exit
EOT
print encode_utf8($message);
exit;
}
# Local Variables:
# indent-tabs-mode: nil
# cperl-indent-level: 4
# End:
# vim: syntax=perl sw=4 sts=4 sr et
|