4 use Fcntl qw/O_RDONLY/;
6 $config_path = '/etc/debbugs';
7 $lib_path = '/usr/lib/debbugs';
8 require "$lib_path/errorlib";
10 my $common_archive = 0;
11 my $common_repeatmerged = 1;
12 my %common_include = ();
13 my %common_exclude = ();
14 my $common_raw_sort = 0;
15 my $common_bug_reverse = 0;
16 my $common_pending_reverse = 0;
17 my $common_severity_reverse = 0;
19 my @common_pending_include = ();
20 my @common_pending_exclude = ();
21 my @common_severity_include = ();
22 my @common_severity_exclude = ();
28 if ($opt eq "archive") { $common_archive = $val; }
29 if ($opt eq "repeatmerged") { $common_repeatmerged = $val; }
30 if ($opt eq "exclude") {
32 @vals = ( $val ) if (ref($val) eq "" && $val );
33 @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
34 @vals = @{$val} if (ref($val) eq "ARRAY" );
35 %common_exclude = map {
36 if (/^(.*):(.*)$/) { ($1, $2) } else { ($_, 1) }
37 } split /[\s,]+/, join ',', @vals;
39 if ($opt eq "include") {
41 @vals = ( $val, ) if (ref($val) eq "" && $val );
42 @vals = ( $$val, ) if (ref($val) eq "SCALAR" && $$val );
43 @vals = @{$val} if (ref($val) eq "ARRAY" );
44 %common_include = map {
45 if (/^(.*):(.*)$/) { ($1, $2) } else { ($_, 1) }
46 } split /[\s,]+/, join ',', @vals;
48 if ($opt eq "raw") { $common_raw_sort = $val; }
49 if ($opt eq "bug-rev") { $common_bug_reverse = $val; }
50 if ($opt eq "pend-rev") { $common_pending_reverse = $val; }
51 if ($opt eq "sev-rev") { $common_severity_reverse = $val; }
52 if ($opt eq "pend-exc") {
54 @vals = ( $val ) if (ref($val) eq "" && $val );
55 @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
56 @vals = @{$val} if (ref($val) eq "ARRAY" );
57 @common_pending_exclude = @vals if (@vals);
59 if ($opt eq "pend-inc") {
61 @vals = ( $val, ) if (ref($val) eq "" && $val );
62 @vals = ( $$val, ) if (ref($val) eq "SCALAR" && $$val );
63 @vals = @{$val} if (ref($val) eq "ARRAY" );
64 @common_pending_include = @vals if (@vals);
66 if ($opt eq "sev-exc") {
68 @vals = ( $val ) if (ref($val) eq "" && $val );
69 @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
70 @vals = @{$val} if (ref($val) eq "ARRAY" );
71 @common_severity_exclude = @vals if (@vals);
73 if ($opt eq "sev-inc") {
75 @vals = ( $val ) if (ref($val) eq "" && $val );
76 @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
77 @vals = @{$val} if (ref($val) eq "ARRAY" );
78 @common_severity_include = @vals if (@vals);
83 my ($in, $key, $val, %ret);
84 if (defined $ENV{"QUERY_STRING"} && $ENV{"QUERY_STRING"} ne "") {
85 $in=$ENV{QUERY_STRING};
86 } elsif(defined $ENV{"REQUEST_METHOD"}
87 && $ENV{"REQUEST_METHOD"} eq "POST")
89 read(STDIN,$in,$ENV{CONTENT_LENGTH});
93 foreach (split(/[&;]/,$in)) {
95 ($key, $val) = split(/=/,$_,2);
96 $key=~s/%(..)/pack("c",hex($1))/ge;
97 $val=~s/%(..)/pack("c",hex($1))/ge;
98 if ( exists $ret{$key} ) {
99 if ( !exists $ret{"&$key"} ) {
100 $ret{"&$key"} = [ $ret{$key} ];
102 push @{$ret{"&$key"}},$val;
106 $debug = 1 if (defined $ret{"debug"} && $ret{"debug"} eq "aj");
112 print "Content-Type: text/html\n\n";
113 print "<HTML><HEAD><TITLE>Error</TITLE></HEAD><BODY>\n";
114 print "An error occurred. Dammit.\n";
115 print "Error was: $msg.\n";
116 print "</BODY></HTML>\n";
122 # my $Archive = $common_archive ? "archive" : "";
123 # print header . start_html("Sorry");
124 # print "Sorry bug #$msg doesn't seem to be in the $Archive database.\n";
129 # Split a package string from the status file into a list of package names.
132 return unless defined $pkgs;
133 return map lc, split /[ \t?,()]+/, $pkgs;
139 return () unless defined $addr;
140 return @{$_parsedaddrs{$addr}} if exists $_parsedaddrs{$addr};
141 @{$_parsedaddrs{$addr}} = Mail::Address->parse($addr);
142 return @{$_parsedaddrs{$addr}};
145 # Generate a comma-separated list of HTML links to each package given in
146 # $pkgs. $pkgs may be empty, in which case an empty string is returned, or
147 # it may be a comma-separated list of package names.
148 sub htmlpackagelinks {
150 return unless defined $pkgs and $pkgs ne '';
152 my @pkglist = splitpackages($pkgs);
154 my $openstrong = $strong ? '<strong>' : '';
155 my $closestrong = $strong ? '</strong>' : '';
157 return 'Package' . (@pkglist > 1 ? 's' : '') . ': ' .
160 '<a href="' . pkgurl($_) . '">' .
161 $openstrong . htmlsanit($_) . $closestrong . '</a>'
166 # Generate a comma-separated list of HTML links to each maintainer given in
167 # $maints, which should be a comma-separated list of RFC822 addresses.
169 my ($prefixfunc, $maints) = @_;
170 if (defined $maints and $maints ne '') {
171 my @maintaddrs = getparsedaddrs($maints);
172 my $prefix = (ref $prefixfunc) ? $prefixfunc->(scalar @maintaddrs)
175 join ', ', map { sprintf '<a href="%s">%s</a>',
176 mainturl($_->address),
177 htmlsanit($_->format) || '(unknown)'
180 my $prefix = (ref $prefixfunc) ? $prefixfunc->(1) : $prefixfunc;
181 return sprintf '%s<a href="%s">(unknown)</a>', $prefix, mainturl('');
187 my %status = %{getbugstatus($ref)};
188 return htmlindexentrystatus(%status) if (%status);
192 sub htmlindexentrystatus {
198 if ($status{severity} eq 'normal') {
200 } elsif (grep($status{severity} eq $_, @debbugs::gStrongSeverities)) {
201 $showseverity = "<strong>Severity: $status{severity}</strong>;\n";
203 $showseverity = "Severity: <em>$status{severity}</em>;\n";
206 $result .= htmlpackagelinks($status{"package"}, 1);
207 $result .= $showseverity;
208 $result .= "Reported by: <a href=\"" . submitterurl($status{originator})
209 . "\">" . htmlsanit($status{originator}) . "</a>";
210 $result .= ";\nTags: <strong>"
211 . htmlsanit(join(", ", sort(split(/\s+/, $status{tags}))))
213 if (length($status{tags}));
215 my @merged= split(/ /,$status{mergedwith});
216 my $mseparator= ";\nmerged with ";
217 for my $m (@merged) {
218 $result .= $mseparator."<A href=\"" . bugurl($m) . "\">#$m</A>";
222 if (length($status{done})) {
223 $result .= ";\n<strong>Done:</strong> " . htmlsanit($status{done});
225 if (length($status{forwarded})) {
226 $result .= ";\n<strong>Forwarded</strong> to "
227 . maybelink($status{forwarded});
229 my $daysold = int((time - $status{date}) / 86400); # seconds to days
233 $font = "em" if ($daysold > 30);
234 $font = "strong" if ($daysold > 60);
235 $efont = "</$font>" if ($font);
236 $font = "<$font>" if ($font);
238 my $yearsold = int($daysold / 365);
239 $daysold -= $yearsold * 365;
241 $result .= ";\n $font";
243 push @age, "1 year" if ($yearsold == 1);
244 push @age, "$yearsold years" if ($yearsold > 1);
245 push @age, "1 day" if ($daysold == 1);
246 push @age, "$daysold days" if ($daysold > 1);
247 $result .= join(" and ", @age);
248 $result .= " old$efont";
258 my $ref = shift || "";
259 my $params = "submitter=" . emailfromrfc822($ref);
260 $params .= "&archive=yes" if ($common_archive);
261 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
262 return urlsanit("pkgreport.cgi" . "?" . $params);
266 my $ref = shift || "";
267 my $params = "maint=" . emailfromrfc822($ref);
268 $params .= "&archive=yes" if ($common_archive);
269 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
270 return urlsanit("pkgreport.cgi" . "?" . $params);
275 my $params = "pkg=$ref";
276 $params .= "&archive=yes" if ($common_archive);
277 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
279 return urlsanit("pkgreport.cgi" . "?" . "$params");
284 my $params = "src=$ref";
285 $params .= "&archive=yes" if ($common_archive);
286 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
287 return urlsanit("pkgreport.cgi" . "?" . "$params");
292 my $params = "tag=$ref";
293 $params .= "&archive=yes" if ($common_archive);
294 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
295 return urlsanit("pkgreport.cgi" . "?" . "$params");
302 my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
303 $url =~ s/([<>&"])/\&$saniarray{$1};/g;
308 my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
309 my $in = shift || "";
310 $in =~ s/([<>&"])/\&$saniarray{$1};/g;
316 if ($in =~ /^[a-zA-Z0-9+.-]+:/) { # RFC 1738 scheme
317 return qq{<a href="$in">} . htmlsanit($in) . '</a>';
319 return htmlsanit($in);
325 my $params = "bug=$ref";
326 foreach my $val (@_) {
327 $params .= "\&msg=$1" if ($val =~ /^msg=([0-9]+)/);
328 $params .= "\&archive=yes" if (!$common_archive && $val =~ /^archive.*$/);
330 $params .= "&archive=yes" if ($common_archive);
331 $params .= "&repeatmerged=no" unless ($common_repeatmerged);
333 return urlsanit("bugreport.cgi" . "?" . "$params");
338 my $params = "bug=$ref";
340 foreach my $val (@_) {
341 $params .= "\&$1=$2" if ($val =~ /^(msg|att)=([0-9]+)/);
342 $filename = $1 if ($val =~ /^filename=(.*)$/);
344 $params .= "&archive=yes" if ($common_archive);
346 $pathinfo = "/$filename" if $filename ne '';
348 return urlsanit("bugreport.cgi$pathinfo?$params");
353 return urlsanit("bugreport.cgi" . "?" . "bug=$ref&mbox=yes");
357 return @{getbugs(sub { 1 })};
367 my %displayshowpending = ("pending", "outstanding",
368 "pending-fixed", "pending upload",
369 "fixed", "fixed in NMU",
371 "forwarded", "forwarded to upstream software authors");
374 return "<HR><H2>No reports found!</H2></HR>\n";
377 if ( $common_bug_reverse ) {
378 @bugs = sort {$b<=>$a} @bugs;
380 @bugs = sort {$a<=>$b} @bugs;
383 foreach my $bug (@bugs) {
384 my %status = %{getbugstatus($bug)};
386 if (%common_include) {
388 foreach my $t (split /\s+/, $status{tags}) {
389 $okay = 1, last if (defined $common_include{$t});
391 if (defined $common_include{subj}) {
392 if (index($status{subject}, $common_include{subj}) > -1) {
398 if (%common_exclude) {
400 foreach my $t (split /\s+/, $status{tags}) {
401 $okay = 0, last if (defined $common_exclude{$t});
403 if (defined $common_exclude{subj}) {
404 if (index($status{subject}, $common_exclude{subj}) > -1) {
410 next if @common_pending_include and
411 not grep { $_ eq $status{pending} } @common_pending_include;
412 next if @common_severity_include and
413 not grep { $_ eq $status{severity} } @common_severity_include;
414 next if grep { $_ eq $status{pending} } @common_pending_exclude;
415 next if grep { $_ eq $status{severity} } @common_severity_exclude;
417 my @merged = sort {$a<=>$b} ($bug, split(/ /, $status{mergedwith}));
418 next unless ($common_repeatmerged || !$seenmerged{$merged[0]});
419 $seenmerged{$merged[0]} = 1;
421 my $html = sprintf "<li><a href=\"%s\">#%d: %s</a>\n<br>",
422 bugurl($bug), $bug, htmlsanit($status{subject});
423 $html .= htmlindexentrystatus(\%status) . "\n";
424 $section{$status{pending} . "_" . $status{severity}} .= $html;
425 push @rawsort, $html if $common_raw_sort;
430 if ($common_raw_sort) {
431 $result .= "<UL>\n" . join("", @rawsort ) . "</UL>\n";
433 my @pendingList = qw(pending forwarded pending-fixed fixed done);
434 @pendingList = reverse @pendingList if $common_pending_reverse;
435 #print STDERR join(",",@pendingList)."\n";
436 #print STDERR join(",",@common_pending_include).":$#common_pending_include\n";
437 foreach my $pending (@pendingList) {
438 my @severityList = @debbugs::gSeverityList;
439 @severityList = reverse @severityList if $common_severity_reverse;
440 #print STDERR join(",",@severityList)."\n";
442 # foreach my $severity(@debbugs::gSeverityList) {
443 foreach my $severity(@severityList) {
444 $severity = $debbugs::gDefaultSeverity if ($severity eq '');
445 next unless defined $section{${pending} . "_" . ${severity}};
446 $result .= "<HR><H2>$debbugs::gSeverityDisplay{$severity} - $displayshowpending{$pending}</H2>\n";
447 #$result .= "(A list of <a href=\"http://${debbugs::gWebDomain}/db/si/$pending$severity\">all such bugs</a> is available).\n";
448 #$result .= "(A list of all such bugs used to be available).\n";
450 $result .= $section{$pending . "_" . $severity};
451 $result .= "</UL>\n";
452 $anydone = 1 if ($pending eq "done");
457 $result .= $debbugs::gHTMLExpireNote if $gRemoveAge and $anydone;
463 if ($common_archive) {
464 open I, "<$debbugs::gSpoolDir/index.archive"
465 or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
467 open I, "<$debbugs::gSpoolDir/index.db"
468 or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
474 if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
475 my @x = $bugfunc->(pkg => $1, bug => $2, status => $4,
476 submitter => $5, severity => $6, tags => $7);
478 $count{$_}++ foreach @x;
491 if (!$common_archive && defined $opt &&
492 -e "$debbugs::gSpoolDir/by-$opt.idx")
495 print STDERR "optimized\n" if ($debug);
496 tie %lookup, DB_File => "$debbugs::gSpoolDir/by-$opt.idx", O_RDONLY
497 or die "$0: can't open $debbugs::gSpoolDir/by-$opt.idx ($!)\n";
498 while ($key = shift) {
499 my $bugs = $lookup{$key};
501 push @result, (unpack 'N*', $bugs);
505 print STDERR "done optimized\n" if ($debug);
507 if ( $common_archive ) {
508 open I, "<$debbugs::gSpoolDir/index.archive"
509 or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
511 open I, "<$debbugs::gSpoolDir/index.db"
512 or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
515 if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
516 if ($bugfunc->(pkg => $1, bug => $2, status => $4,
517 submitter => $5, severity => $6, tags => $7))
525 @result = sort {$a <=> $b} @result;
529 sub emailfromrfc822 {
531 $email =~ s/\s*\(.*\)\s*//;
532 $email = $1 if ($email =~ m/<(.*)>/);
540 while ($input =~ m/\W/) {
541 $encoded.=$`.sprintf("-%02x_",unpack("C",$&));
546 $encoded =~ s/-2e_/\./g;
547 $encoded =~ s/^([^,]+)-20_-3c_(.*)-40_(.*)-3e_/$1,$2,$3,/;
548 $encoded =~ s/^(.*)-40_(.*)-20_-28_([^,]+)-29_$/,$1,$2,$3/;
549 $encoded =~ s/-20_/_/g;
550 $encoded =~ s/-([^_]+)_-/-$1/g;
556 return $_maintainer if $_maintainer;
559 open(MM,"$gMaintainerFile") or &quitcgi("open $gMaintainerFile: $!");
561 next unless m/^(\S+)\s+(\S.*\S)\s*$/;
567 if (defined $gMaintainerFileOverride) {
568 open(MM,"$gMaintainerFileOverride") or &quitcgi("open $gMaintainerFileOverride: $!");
570 next unless m/^(\S+)\s+(\S.*\S)\s*$/;
577 $_maintainer = \%maintainer;
584 return $_pkgsrc if $_pkgsrc;
585 return {} unless defined $gPackageSource;
589 open(MM,"$gPackageSource") or &quitcgi("open $gPackageSource: $!");
591 next unless m/^(\S+)\s+(\S+)\s+(\S.*\S)\s*$/;
592 ($a,$b,$c)=($1,$2,$3);
595 $pkgcomponent{$a}= $b;
599 $_pkgcomponent = \%pkgcomponent;
603 sub getpkgcomponent {
604 return $_pkgcomponent if $_pkgcomponent;
606 return $_pkgcomponent;
611 return $_pseudodesc if $_pseudodesc;
614 open(PSEUDO, "< $gPseudoDescFile") or &quitcgi("open $gPseudoDescFile: $!");
616 next unless m/^(\S+)\s+(\S.*\S)\s*$/;
617 $pseudodesc{lc $1} = $2;
620 $_pseudodesc = \%pseudodesc;
629 my $location = getbuglocation( $bugnum, "status" );
630 return {} if ( !$location );
631 %status = %{ readbug( $bugnum, $location ) };
633 $status{tags} = $status{keywords};
635 $status{"package"} =~ s/\s*$//;
636 $status{"package"} = 'unknown' if ($status{"package"} eq '');
637 $status{"severity"} = 'normal' if ($status{"severity"} eq '');
639 $status{"pending"} = 'pending';
640 $status{"pending"} = 'forwarded' if (length($status{"forwarded"}));
641 $status{"pending"} = 'pending-fixed' if ($status{"tags"} =~ /\bpending\b/);
642 $status{"pending"} = 'fixed' if ($status{"tags"} =~ /\bfixed\b/);
643 $status{"pending"} = 'done' if (length($status{"done"}));
651 my %pkgsrc = %{getpkgsrc()};
653 foreach ( keys %pkgsrc ) {
654 push @pkgs, $_ if $pkgsrc{$_} eq $src;
661 my $location = getbuglocation($bugnum, 'log');
662 return getbugcomponent($bugnum, 'log', $location);