]> git.donarmstrong.com Git - debbugs.git/blob - cgi/common.pl
[project @ 2003-08-06 23:35:55 by cjwatson]
[debbugs.git] / cgi / common.pl
1 #!/usr/bin/perl -w
2
3 use DB_File;
4 use Fcntl qw/O_RDONLY/;
5 use Mail::Address;
6 $config_path = '/etc/debbugs';
7 $lib_path = '/usr/lib/debbugs';
8 require "$lib_path/errorlib";
9
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;
18
19 my @common_pending_include = ();
20 my @common_pending_exclude = ();
21 my @common_severity_include = ();
22 my @common_severity_exclude = ();
23
24 my $debug = 0;
25
26 sub set_option {
27     my ($opt, $val) = @_;
28     if ($opt eq "archive") { $common_archive = $val; }
29     if ($opt eq "repeatmerged") { $common_repeatmerged = $val; }
30     if ($opt eq "exclude") {
31         my @vals;
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;
38     }
39     if ($opt eq "include") {
40         my @vals;
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;
47     }
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") {
53         my @vals;
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);
58     }
59     if ($opt eq "pend-inc") {
60         my @vals;
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);
65     }
66     if ($opt eq "sev-exc") {
67         my @vals;
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);
72     }
73     if ($opt eq "sev-inc") {
74         my @vals;
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);
79     }
80 }
81
82 sub readparse {
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")
88     {
89         read(STDIN,$in,$ENV{CONTENT_LENGTH});
90     } else {
91         return;
92     }
93     foreach (split(/[&;]/,$in)) {
94         s/\+/ /g;
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} ];
101             }
102             push @{$ret{"&$key"}},$val;
103         }
104         $ret{$key}=$val;
105     }
106 $debug = 1 if (defined $ret{"debug"} && $ret{"debug"} eq "aj");
107     return %ret;
108 }
109
110 sub quitcgi {
111     my $msg = shift;
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";
117     exit 0;
118 }
119
120 #sub abort {
121 #    my $msg = shift;
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";
125 #    print end_html;
126 #    exit 0;
127 #}
128
129 # Split a package string from the status file into a list of package names.
130 sub splitpackages {
131     my $pkgs = shift;
132     return unless defined $pkgs;
133     return map lc, split /[ \t?,()]+/, $pkgs;
134 }
135
136 my %_parsedaddrs;
137 sub getparsedaddrs {
138     my $addr = shift;
139     return () unless defined $addr;
140     return @{$_parsedaddrs{$addr}} if exists $_parsedaddrs{$addr};
141     @{$_parsedaddrs{$addr}} = Mail::Address->parse($addr);
142     return @{$_parsedaddrs{$addr}};
143 }
144
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 {
149     my $pkgs = shift;
150     return unless defined $pkgs and $pkgs ne '';
151     my $strong = shift;
152     my @pkglist = splitpackages($pkgs);
153
154     my $openstrong  = $strong ? '<strong>' : '';
155     my $closestrong = $strong ? '</strong>' : '';
156
157     return 'Package' . (@pkglist > 1 ? 's' : '') . ': ' .
158            join(', ',
159                 map {
160                     '<a href="' . pkgurl($_) . '">' .
161                     $openstrong . htmlsanit($_) . $closestrong . '</a>'
162                 } @pkglist
163            ) . ";\n";
164 }
165
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.
168 sub htmlmaintlinks {
169     my ($prefixfunc, $maints) = @_;
170     if (defined $maints and $maints ne '') {
171         my @maintaddrs = getparsedaddrs($maints);
172         my $prefix = (ref $prefixfunc) ? $prefixfunc->(scalar @maintaddrs)
173                                        : $prefixfunc;
174         return $prefix .
175                join ', ', map { sprintf '<a href="%s">%s</a>',
176                                         mainturl($_->address),
177                                         htmlsanit($_->format) || '(unknown)'
178                               } @maintaddrs;
179     } else {
180         my $prefix = (ref $prefixfunc) ? $prefixfunc->(1) : $prefixfunc;
181         return sprintf '%s<a href="%s">(unknown)</a>', $prefix, mainturl('');
182     }
183 }
184
185 sub htmlindexentry {
186     my $ref = shift;
187     my %status = %{getbugstatus($ref)};
188     return htmlindexentrystatus(%status) if (%status);
189     return "";
190 }
191
192 sub htmlindexentrystatus {
193     my $s = shift;
194     my %status = %{$s};
195
196     my $result = "";
197
198     if  ($status{severity} eq 'normal') {
199         $showseverity = '';
200     } elsif (grep($status{severity} eq $_, @debbugs::gStrongSeverities)) {
201         $showseverity = "<strong>Severity: $status{severity}</strong>;\n";
202     } else {
203         $showseverity = "Severity: <em>$status{severity}</em>;\n";
204     }
205
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}))))
212                  . "</strong>"
213                        if (length($status{tags}));
214
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>";
219         $mseparator= ", ";
220     }
221
222     if (length($status{done})) {
223         $result .= ";\n<strong>Done:</strong> " . htmlsanit($status{done});
224     } else {
225         if (length($status{forwarded})) {
226             $result .= ";\n<strong>Forwarded</strong> to "
227                        . maybelink($status{forwarded});
228         }
229         my $daysold = int((time - $status{date}) / 86400);   # seconds to days
230         if ($daysold >= 7) {
231             my $font = "";
232             my $efont = "";
233             $font = "em" if ($daysold > 30);
234             $font = "strong" if ($daysold > 60);
235             $efont = "</$font>" if ($font);
236             $font = "<$font>" if ($font);
237
238             my $yearsold = int($daysold / 365);
239             $daysold -= $yearsold * 365;
240
241             $result .= ";\n $font";
242             my @age;
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";
249         }
250     }
251
252     $result .= ".";
253
254     return $result;
255 }
256
257 sub submitterurl {
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);
263 }
264
265 sub mainturl {
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);
271 }
272
273 sub pkgurl {
274     my $ref = shift;
275     my $params = "pkg=$ref";
276     $params .= "&archive=yes" if ($common_archive);
277     $params .= "&repeatmerged=no" unless ($common_repeatmerged);
278     
279     return urlsanit("pkgreport.cgi" . "?" . "$params");
280 }
281
282 sub srcurl {
283     my $ref = shift;
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");
288 }
289
290 sub tagurl {
291     my $ref = shift;
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");
296 }
297
298 sub urlsanit {
299     my $url = shift;
300     $url =~ s/%/%25/g;
301     $url =~ s/\+/%2b/g;
302     my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
303     $url =~ s/([<>&"])/\&$saniarray{$1};/g;
304     return $url;
305 }
306
307 sub htmlsanit {
308     my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
309     my $in = shift || "";
310     $in =~ s/([<>&"])/\&$saniarray{$1};/g;
311     return $in;
312 }
313
314 sub maybelink {
315     my $in = shift;
316     if ($in =~ /^[a-zA-Z0-9+.-]+:/) { # RFC 1738 scheme
317         return qq{<a href="$in">} . htmlsanit($in) . '</a>';
318     } else {
319         return htmlsanit($in);
320     }
321 }
322
323 sub bugurl {
324     my $ref = shift;
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.*$/);
329     }
330     $params .= "&archive=yes" if ($common_archive);
331     $params .= "&repeatmerged=no" unless ($common_repeatmerged);
332
333     return urlsanit("bugreport.cgi" . "?" . "$params");
334 }
335
336 sub dlurl {
337     my $ref = shift;
338     my $params = "bug=$ref";
339     my $filename = '';
340     foreach my $val (@_) {
341         $params .= "\&$1=$2" if ($val =~ /^(msg|att)=([0-9]+)/);
342         $filename = $1 if ($val =~ /^filename=(.*)$/);
343     }
344     $params .= "&archive=yes" if ($common_archive);
345     my $pathinfo = '';
346     $pathinfo = "/$filename" if $filename ne '';
347
348     return urlsanit("bugreport.cgi$pathinfo?$params");
349 }
350
351 sub mboxurl {
352     my $ref = shift;
353     return urlsanit("bugreport.cgi" . "?" . "bug=$ref&mbox=yes");
354 }
355
356 sub allbugs {
357     return @{getbugs(sub { 1 })};
358 }
359
360 sub htmlizebugs {
361     $b = $_[0];
362     my @bugs = @$b;
363     my @rawsort;
364
365     my %section = ();
366
367     my %displayshowpending = ("pending", "outstanding",
368                               "pending-fixed", "pending upload",
369                               "fixed", "fixed in NMU",
370                               "done", "resolved",
371                               "forwarded", "forwarded to upstream software authors");
372
373     if (@bugs == 0) {
374         return "<HR><H2>No reports found!</H2></HR>\n";
375     }
376
377     if ( $common_bug_reverse ) {
378         @bugs = sort {$b<=>$a} @bugs;
379     } else {
380         @bugs = sort {$a<=>$b} @bugs;
381     }
382     my %seenmerged;
383     foreach my $bug (@bugs) {
384         my %status = %{getbugstatus($bug)};
385         next unless %status;
386         if (%common_include) {
387             my $okay = 0;
388             foreach my $t (split /\s+/, $status{tags}) {
389                 $okay = 1, last if (defined $common_include{$t});
390             }
391             if (defined $common_include{subj}) {
392                 if (index($status{subject}, $common_include{subj}) > -1) {
393                     $okay = 1;
394                 }
395             }
396             next unless ($okay);
397         }
398         if (%common_exclude) {
399             my $okay = 1;
400             foreach my $t (split /\s+/, $status{tags}) {
401                 $okay = 0, last if (defined $common_exclude{$t});
402             }
403             if (defined $common_exclude{subj}) {
404                 if (index($status{subject}, $common_exclude{subj}) > -1) {
405                     $okay = 0;
406                 }
407             }
408             next unless ($okay);
409         }
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;
416
417         my @merged = sort {$a<=>$b} ($bug, split(/ /, $status{mergedwith}));
418         next unless ($common_repeatmerged || !$seenmerged{$merged[0]});
419         $seenmerged{$merged[0]} = 1;
420
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;
426     }
427
428     my $result = "";
429     my $anydone = 0;
430     if ($common_raw_sort) {
431         $result .= "<UL>\n" . join("", @rawsort ) . "</UL>\n";
432     } else {
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";
441
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";
449             $result .= "<UL>\n";
450             $result .= $section{$pending . "_" . $severity}; 
451             $result .= "</UL>\n";
452             $anydone = 1 if ($pending eq "done");
453          }
454     }
455
456     }
457     $result .= $debbugs::gHTMLExpireNote if $gRemoveAge and $anydone;
458     return $result;
459 }
460
461 sub countbugs {
462     my $bugfunc = shift;
463     if ($common_archive) {
464         open I, "<$debbugs::gSpoolDir/index.archive"
465             or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
466     } else {
467         open I, "<$debbugs::gSpoolDir/index.db"
468             or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
469     }
470
471     my %count = ();
472     while(<I>) 
473     {
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);
477             local $_;
478             $count{$_}++ foreach @x;
479         }
480     }
481     close I;
482     return %count;
483 }
484
485 sub getbugs {
486     my $bugfunc = shift;
487     my $opt = shift;
488
489     my @result = ();
490
491     if (!$common_archive && defined $opt && 
492         -e "$debbugs::gSpoolDir/by-$opt.idx") 
493     {
494         my %lookup;
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};
500             if (defined $bugs) {
501                 push @result, (unpack 'N*', $bugs);
502             }
503         }
504         untie %lookup;
505 print STDERR "done optimized\n" if ($debug);
506     } else {
507         if ( $common_archive ) {
508             open I, "<$debbugs::gSpoolDir/index.archive" 
509                 or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
510         } else {
511             open I, "<$debbugs::gSpoolDir/index.db" 
512                 or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
513         }
514         while(<I>) {
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)) 
518                 {
519                     push (@result, $2);
520                 }
521             }
522         }
523         close I;
524     }
525     @result = sort {$a <=> $b} @result;
526     return \@result;
527 }
528
529 sub emailfromrfc822 {
530     my $email = shift;
531     $email =~ s/\s*\(.*\)\s*//;
532     $email = $1 if ($email =~ m/<(.*)>/);
533     return $email;
534 }
535
536 sub maintencoded {
537     my $input = shift;
538     my $encoded = '';
539
540     while ($input =~ m/\W/) {
541         $encoded.=$`.sprintf("-%02x_",unpack("C",$&));
542         $input= $';
543     }
544
545     $encoded.= $input;
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;
551     return $encoded;
552 }
553
554 my $_maintainer;
555 sub getmaintainers {
556     return $_maintainer if $_maintainer;
557     my %maintainer;
558
559     open(MM,"$gMaintainerFile") or &quitcgi("open $gMaintainerFile: $!");
560     while(<MM>) {
561         next unless m/^(\S+)\s+(\S.*\S)\s*$/;
562         ($a,$b)=($1,$2);
563         $a =~ y/A-Z/a-z/;
564         $maintainer{$a}= $b;
565     }
566     close(MM);
567     if (defined $gMaintainerFileOverride) {
568         open(MM,"$gMaintainerFileOverride") or &quitcgi("open $gMaintainerFileOverride: $!");
569         while(<MM>) {
570             next unless m/^(\S+)\s+(\S.*\S)\s*$/;
571             ($a,$b)=($1,$2);
572             $a =~ y/A-Z/a-z/;
573             $maintainer{$a}= $b;
574         }
575         close(MM);
576     }
577     $_maintainer = \%maintainer;
578     return $_maintainer;
579 }
580
581 my $_pkgsrc;
582 my $_pkgcomponent;
583 sub getpkgsrc {
584     return $_pkgsrc if $_pkgsrc;
585     return {} unless defined $gPackageSource;
586     my %pkgsrc;
587     my %pkgcomponent;
588
589     open(MM,"$gPackageSource") or &quitcgi("open $gPackageSource: $!");
590     while(<MM>) {
591         next unless m/^(\S+)\s+(\S+)\s+(\S.*\S)\s*$/;
592         ($a,$b,$c)=($1,$2,$3);
593         $a =~ y/A-Z/a-z/;
594         $pkgsrc{$a}= $c;
595         $pkgcomponent{$a}= $b;
596     }
597     close(MM);
598     $_pkgsrc = \%pkgsrc;
599     $_pkgcomponent = \%pkgcomponent;
600     return $_pkgsrc;
601 }
602
603 sub getpkgcomponent {
604     return $_pkgcomponent if $_pkgcomponent;
605     getpkgsrc();
606     return $_pkgcomponent;
607 }
608
609 my $_pseudodesc;
610 sub getpseudodesc {
611     return $_pseudodesc if $_pseudodesc;
612     my %pseudodesc;
613
614     open(PSEUDO, "< $gPseudoDescFile") or &quitcgi("open $gPseudoDescFile: $!");
615     while(<PSEUDO>) {
616         next unless m/^(\S+)\s+(\S.*\S)\s*$/;
617         $pseudodesc{lc $1} = $2;
618     }
619     close(PSEUDO);
620     $_pseudodesc = \%pseudodesc;
621     return $_pseudodesc;
622 }
623
624 sub getbugstatus {
625     my $bugnum = shift;
626
627     my %status;
628
629     my $location = getbuglocation( $bugnum, "status" );
630     return {} if ( !$location );
631     %status = %{ readbug( $bugnum, $location ) };
632
633     $status{tags} = $status{keywords};
634
635     $status{"package"} =~ s/\s*$//;
636     $status{"package"} = 'unknown' if ($status{"package"} eq '');
637     $status{"severity"} = 'normal' if ($status{"severity"} eq '');
638
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"}));
644
645     return \%status;
646 }
647
648 sub getsrcpkgs {
649     my $src = shift;
650     return () if !$src;
651     my %pkgsrc = %{getpkgsrc()};
652     my @pkgs;
653     foreach ( keys %pkgsrc ) {
654         push @pkgs, $_ if $pkgsrc{$_} eq $src;
655     }
656     return @pkgs;
657 }
658    
659 sub buglog {
660     my $bugnum = shift;
661     my $location = getbuglocation($bugnum, 'log');
662     return getbugcomponent($bugnum, 'log', $location);
663 }
664
665 1;