]> git.donarmstrong.com Git - debbugs.git/blob - cgi/common.pl
[project @ 2003-09-15 14:23:45 by doogie]
[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 use MLDBM qw/DB_File/;
7
8 $config_path = '/etc/debbugs';
9 $lib_path = '/usr/lib/debbugs';
10 require "$lib_path/errorlib";
11
12 use Debbugs::Versions;
13
14 $MLDBM::RemoveTaint = 1;
15
16 my $common_archive = 0;
17 my $common_repeatmerged = 1;
18 my %common_include = ();
19 my %common_exclude = ();
20 my $common_raw_sort = 0;
21 my $common_bug_reverse = 0;
22 my $common_pending_reverse = 0;
23 my $common_severity_reverse = 0;
24
25 my @common_pending_include = ();
26 my @common_pending_exclude = ();
27 my @common_severity_include = ();
28 my @common_severity_exclude = ();
29
30 my $common_version;
31 my $common_dist;
32 my $common_arch;
33
34 my $debug = 0;
35
36 sub set_option {
37     my ($opt, $val) = @_;
38     if ($opt eq "archive") { $common_archive = $val; }
39     if ($opt eq "repeatmerged") { $common_repeatmerged = $val; }
40     if ($opt eq "exclude") {
41         my @vals;
42         @vals = ( $val ) if (ref($val) eq "" && $val );
43         @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
44         @vals = @{$val} if (ref($val) eq "ARRAY" );
45         %common_exclude = map {
46             if (/^([^:]*):(.*)$/) { ($1, $2) } else { ($_, 1) }
47         } split /[\s,]+/, join ',', @vals;
48     }
49     if ($opt eq "include") {
50         my @vals;
51         @vals = ( $val, ) if (ref($val) eq "" && $val );
52         @vals = ( $$val, ) if (ref($val) eq "SCALAR" && $$val );
53         @vals = @{$val} if (ref($val) eq "ARRAY" );
54         %common_include = map {
55             if (/^([^:]*):(.*)$/) { ($1, $2) } else { ($_, 1) }
56         } split /[\s,]+/, join ',', @vals;
57     }
58     if ($opt eq "raw") { $common_raw_sort = $val; }
59     if ($opt eq "bug-rev") { $common_bug_reverse = $val; }
60     if ($opt eq "pend-rev") { $common_pending_reverse = $val; }
61     if ($opt eq "sev-rev") { $common_severity_reverse = $val; }
62     if ($opt eq "pend-exc") {
63         my @vals;
64         @vals = ( $val ) if (ref($val) eq "" && $val );
65         @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
66         @vals = @{$val} if (ref($val) eq "ARRAY" );
67         @common_pending_exclude = @vals if (@vals);
68     }
69     if ($opt eq "pend-inc") {
70         my @vals;
71         @vals = ( $val, ) if (ref($val) eq "" && $val );
72         @vals = ( $$val, ) if (ref($val) eq "SCALAR" && $$val );
73         @vals = @{$val} if (ref($val) eq "ARRAY" );
74         @common_pending_include = @vals if (@vals);
75     }
76     if ($opt eq "sev-exc") {
77         my @vals;
78         @vals = ( $val ) if (ref($val) eq "" && $val );
79         @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
80         @vals = @{$val} if (ref($val) eq "ARRAY" );
81         @common_severity_exclude = @vals if (@vals);
82     }
83     if ($opt eq "sev-inc") {
84         my @vals;
85         @vals = ( $val ) if (ref($val) eq "" && $val );
86         @vals = ( $$val ) if (ref($val) eq "SCALAR" && $$val );
87         @vals = @{$val} if (ref($val) eq "ARRAY" );
88         @common_severity_include = @vals if (@vals);
89     }
90     if ($opt eq "version") { $common_version = $val; }
91     if ($opt eq "dist") { $common_dist = $val; }
92     if ($opt eq "arch") { $common_arch = $val; }
93 }
94
95 sub readparse {
96     my ($in, $key, $val, %ret);
97     if (defined $ENV{"QUERY_STRING"} && $ENV{"QUERY_STRING"} ne "") {
98         $in=$ENV{QUERY_STRING};
99     } elsif(defined $ENV{"REQUEST_METHOD"}
100         && $ENV{"REQUEST_METHOD"} eq "POST")
101     {
102         read(STDIN,$in,$ENV{CONTENT_LENGTH});
103     } else {
104         return;
105     }
106     foreach (split(/[&;]/,$in)) {
107         s/\+/ /g;
108         ($key, $val) = split(/=/,$_,2);
109         $key=~s/%(..)/pack("c",hex($1))/ge;
110         $val=~s/%(..)/pack("c",hex($1))/ge;
111         if ( exists $ret{$key} ) {
112             if ( !exists $ret{"&$key"} ) {
113                 $ret{"&$key"} = [ $ret{$key} ];
114             }
115             push @{$ret{"&$key"}},$val;
116         }
117         $ret{$key}=$val;
118     }
119 $debug = 1 if (defined $ret{"debug"} && $ret{"debug"} eq "aj");
120     return %ret;
121 }
122
123 sub quitcgi {
124     my $msg = shift;
125     print "Content-Type: text/html\n\n";
126     print "<HTML><HEAD><TITLE>Error</TITLE></HEAD><BODY>\n";
127     print "An error occurred. Dammit.\n";
128     print "Error was: $msg.\n";
129     print "</BODY></HTML>\n";
130     exit 0;
131 }
132
133 #sub abort {
134 #    my $msg = shift;
135 #    my $Archive = $common_archive ? "archive" : "";
136 #    print header . start_html("Sorry");
137 #    print "Sorry bug #$msg doesn't seem to be in the $Archive database.\n";
138 #    print end_html;
139 #    exit 0;
140 #}
141
142 # Split a package string from the status file into a list of package names.
143 sub splitpackages {
144     my $pkgs = shift;
145     return unless defined $pkgs;
146     return map lc, split /[ \t?,()]+/, $pkgs;
147 }
148
149 my %_parsedaddrs;
150 sub getparsedaddrs {
151     my $addr = shift;
152     return () unless defined $addr;
153     return @{$_parsedaddrs{$addr}} if exists $_parsedaddrs{$addr};
154     @{$_parsedaddrs{$addr}} = Mail::Address->parse($addr);
155     return @{$_parsedaddrs{$addr}};
156 }
157
158 # Generate a comma-separated list of HTML links to each package given in
159 # $pkgs. $pkgs may be empty, in which case an empty string is returned, or
160 # it may be a comma-separated list of package names.
161 sub htmlpackagelinks {
162     my $pkgs = shift;
163     return unless defined $pkgs and $pkgs ne '';
164     my $strong = shift;
165     my @pkglist = splitpackages($pkgs);
166
167     my $openstrong  = $strong ? '<strong>' : '';
168     my $closestrong = $strong ? '</strong>' : '';
169
170     return 'Package' . (@pkglist > 1 ? 's' : '') . ': ' .
171            join(', ',
172                 map {
173                     '<a href="' . pkgurl($_) . '">' .
174                     $openstrong . htmlsanit($_) . $closestrong . '</a>'
175                 } @pkglist
176            ) . ";\n";
177 }
178
179 # Generate a comma-separated list of HTML links to each maintainer given in
180 # $maints, which should be a comma-separated list of RFC822 addresses.
181 sub htmlmaintlinks {
182     my ($prefixfunc, $maints) = @_;
183     if (defined $maints and $maints ne '') {
184         my @maintaddrs = getparsedaddrs($maints);
185         my $prefix = (ref $prefixfunc) ? $prefixfunc->(scalar @maintaddrs)
186                                        : $prefixfunc;
187         return $prefix .
188                join ', ', map { sprintf '<a href="%s">%s</a>',
189                                         mainturl($_->address),
190                                         htmlsanit($_->format) || '(unknown)'
191                               } @maintaddrs;
192     } else {
193         my $prefix = (ref $prefixfunc) ? $prefixfunc->(1) : $prefixfunc;
194         return sprintf '%s<a href="%s">(unknown)</a>', $prefix, mainturl('');
195     }
196 }
197
198 sub htmlindexentry {
199     my $ref = shift;
200     my %status = %{getbugstatus($ref)};
201     return htmlindexentrystatus(%status) if (%status);
202     return "";
203 }
204
205 sub htmlindexentrystatus {
206     my $s = shift;
207     my %status = %{$s};
208
209     my $result = "";
210
211     if  ($status{severity} eq 'normal') {
212         $showseverity = '';
213     } elsif (isstrongseverity($status{severity})) {
214         $showseverity = "<strong>Severity: $status{severity}</strong>;\n";
215     } else {
216         $showseverity = "Severity: <em>$status{severity}</em>;\n";
217     }
218
219     $result .= htmlpackagelinks($status{"package"}, 1);
220     $result .= $showseverity;
221     $result .= "Reported by: <a href=\"" . submitterurl($status{originator})
222                . "\">" . htmlsanit($status{originator}) . "</a>";
223     $result .= ";\nOwned by: " . htmlsanit($status{owner})
224                if length $status{owner};
225     $result .= ";\nTags: <strong>" 
226                  . htmlsanit(join(", ", sort(split(/\s+/, $status{tags}))))
227                  . "</strong>"
228                        if (length($status{tags}));
229
230     my @merged= split(/ /,$status{mergedwith});
231     my $mseparator= ";\nmerged with ";
232     for my $m (@merged) {
233         $result .= $mseparator."<A href=\"" . bugurl($m) . "\">#$m</A>";
234         $mseparator= ", ";
235     }
236
237     if (@{$status{found_versions}}) {
238         $result .= ";\nfound in ";
239         $result .= (@{$status{found_versions}} == 1) ? 'version '
240                                                      : 'versions ';
241         $result .= join ', ', map htmlsanit($_), @{$status{found_versions}};
242     }
243
244     if (@{$status{fixed_versions}}) {
245         $result .= ";\n<strong>fixed</strong> in ";
246         $result .= (@{$status{fixed_versions}} == 1) ? 'version '
247                                                      : 'versions ';
248         $result .= join ', ', map htmlsanit($_), @{$status{fixed_versions}};
249         if (length($status{done})) {
250             $result .= ' by ' . htmlsanit($status{done});
251         }
252     } elsif (length($status{done})) {
253         $result .= ";\n<strong>Done:</strong> " . htmlsanit($status{done});
254     }
255
256     unless (length($status{done})) {
257         if (length($status{forwarded})) {
258             $result .= ";\n<strong>Forwarded</strong> to "
259                        . maybelink($status{forwarded});
260         }
261         my $daysold = int((time - $status{date}) / 86400);   # seconds to days
262         if ($daysold >= 7) {
263             my $font = "";
264             my $efont = "";
265             $font = "em" if ($daysold > 30);
266             $font = "strong" if ($daysold > 60);
267             $efont = "</$font>" if ($font);
268             $font = "<$font>" if ($font);
269
270             my $yearsold = int($daysold / 365);
271             $daysold -= $yearsold * 365;
272
273             $result .= ";\n $font";
274             my @age;
275             push @age, "1 year" if ($yearsold == 1);
276             push @age, "$yearsold years" if ($yearsold > 1);
277             push @age, "1 day" if ($daysold == 1);
278             push @age, "$daysold days" if ($daysold > 1);
279             $result .= join(" and ", @age);
280             $result .= " old$efont";
281         }
282     }
283
284     $result .= ".";
285
286     return $result;
287 }
288
289 sub urlargs {
290     my $args = '';
291     $args .= "&archive=yes" if $common_archive;
292     $args .= "&repeatmerged=no" unless $common_repeatmerged;
293     $args .= "&version=$common_version" if defined $common_version;
294     $args .= "&dist=$common_dist" if defined $common_dist;
295     $args .= "&arch=$common_arch" if defined $common_arch;
296     return $args;
297 }
298
299 sub submitterurl {
300     my $ref = shift || "";
301     my $params = "submitter=" . emailfromrfc822($ref);
302     $params .= urlargs();
303     return urlsanit("pkgreport.cgi" . "?" . $params);
304 }
305
306 sub mainturl {
307     my $ref = shift || "";
308     my $params = "maint=" . emailfromrfc822($ref);
309     $params .= urlargs();
310     return urlsanit("pkgreport.cgi" . "?" . $params);
311 }
312
313 sub pkgurl {
314     my $ref = shift;
315     my $params = "pkg=$ref";
316     $params .= urlargs();
317     return urlsanit("pkgreport.cgi" . "?" . "$params");
318 }
319
320 sub srcurl {
321     my $ref = shift;
322     my $params = "src=$ref";
323     $params .= urlargs();
324     return urlsanit("pkgreport.cgi" . "?" . "$params");
325 }
326
327 sub tagurl {
328     my $ref = shift;
329     my $params = "tag=$ref";
330     $params .= urlargs();
331     return urlsanit("pkgreport.cgi" . "?" . "$params");
332 }
333
334 sub urlsanit {
335     my $url = shift;
336     $url =~ s/%/%25/g;
337     $url =~ s/\+/%2b/g;
338     my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
339     $url =~ s/([<>&"])/\&$saniarray{$1};/g;
340     return $url;
341 }
342
343 sub htmlsanit {
344     my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
345     my $in = shift || "";
346     $in =~ s/([<>&"])/\&$saniarray{$1};/g;
347     return $in;
348 }
349
350 sub maybelink {
351     my $in = shift;
352     if ($in =~ /^[a-zA-Z0-9+.-]+:/) { # RFC 1738 scheme
353         return qq{<a href="$in">} . htmlsanit($in) . '</a>';
354     } else {
355         return htmlsanit($in);
356     }
357 }
358
359 sub bugurl {
360     my $ref = shift;
361     my $params = "bug=$ref";
362     foreach my $val (@_) {
363         $params .= "\&msg=$1" if ($val =~ /^msg=([0-9]+)/);
364         $params .= "\&archive=yes" if (!$common_archive && $val =~ /^archive.*$/);
365     }
366     $params .= "&archive=yes" if ($common_archive);
367     $params .= "&repeatmerged=no" unless ($common_repeatmerged);
368
369     return urlsanit("bugreport.cgi" . "?" . "$params");
370 }
371
372 sub dlurl {
373     my $ref = shift;
374     my $params = "bug=$ref";
375     my $filename = '';
376     foreach my $val (@_) {
377         $params .= "\&$1=$2" if ($val =~ /^(msg|att)=([0-9]+)/);
378         $filename = $1 if ($val =~ /^filename=(.*)$/);
379     }
380     $params .= "&archive=yes" if ($common_archive);
381     my $pathinfo = '';
382     $pathinfo = "/$filename" if $filename ne '';
383
384     return urlsanit("bugreport.cgi$pathinfo?$params");
385 }
386
387 sub mboxurl {
388     my $ref = shift;
389     return urlsanit("bugreport.cgi" . "?" . "bug=$ref&mbox=yes");
390 }
391
392 sub allbugs {
393     return @{getbugs(sub { 1 })};
394 }
395
396 sub htmlizebugs {
397     $b = $_[0];
398     my @bugs = @$b;
399     my @rawsort;
400
401     my %section = ();
402
403     my %displayshowpending = ("pending", "outstanding",
404                               "pending-fixed", "pending upload",
405                               "fixed", "fixed in NMU",
406                               "done", "resolved",
407                               "forwarded", "forwarded to upstream software authors",
408                               "absent", "not applicable to this version");
409
410     if (@bugs == 0) {
411         return "<HR><H2>No reports found!</H2></HR>\n";
412     }
413
414     if ( $common_bug_reverse ) {
415         @bugs = sort {$b<=>$a} @bugs;
416     } else {
417         @bugs = sort {$a<=>$b} @bugs;
418     }
419     my %seenmerged;
420     foreach my $bug (@bugs) {
421         my %status = %{getbugstatus($bug)};
422         next unless %status;
423         if (%common_include) {
424             my $okay = 0;
425             foreach my $t (split /\s+/, $status{tags}) {
426                 $okay = 1, last if (defined $common_include{$t});
427             }
428             if (defined $common_include{subj}) {
429                 if (index($status{subject}, $common_include{subj}) > -1) {
430                     $okay = 1;
431                 }
432             }
433             next unless ($okay);
434         }
435         if (%common_exclude) {
436             my $okay = 1;
437             foreach my $t (split /\s+/, $status{tags}) {
438                 $okay = 0, last if (defined $common_exclude{$t});
439             }
440             if (defined $common_exclude{subj}) {
441                 if (index($status{subject}, $common_exclude{subj}) > -1) {
442                     $okay = 0;
443                 }
444             }
445             next unless ($okay);
446         }
447         next if @common_pending_include and
448              not grep { $_ eq $status{pending} } @common_pending_include;
449         next if @common_severity_include and
450              not grep { $_ eq $status{severity} } @common_severity_include;
451         next if grep { $_ eq $status{pending} } @common_pending_exclude;
452         next if grep { $_ eq $status{severity} } @common_severity_exclude;
453
454         my @merged = sort {$a<=>$b} ($bug, split(/ /, $status{mergedwith}));
455         next unless ($common_repeatmerged || !$seenmerged{$merged[0]});
456         $seenmerged{$merged[0]} = 1;
457
458         my $html = sprintf "<li><a href=\"%s\">#%d: %s</a>\n<br>",
459             bugurl($bug), $bug, htmlsanit($status{subject});
460         $html .= htmlindexentrystatus(\%status) . "\n";
461         $section{$status{pending} . "_" . $status{severity}} .= $html;
462         push @rawsort, $html if $common_raw_sort;
463     }
464
465     my $result = "";
466     my $anydone = 0;
467     if ($common_raw_sort) {
468         $result .= "<UL>\n" . join("", @rawsort ) . "</UL>\n";
469     } else {
470         my @pendingList = qw(pending forwarded pending-fixed fixed done absent);
471         @pendingList = reverse @pendingList if $common_pending_reverse;
472 #print STDERR join(",",@pendingList)."\n";
473 #print STDERR join(",",@common_pending_include).":$#common_pending_include\n";
474     foreach my $pending (@pendingList) {
475         my @severityList = @debbugs::gSeverityList;
476         @severityList = reverse @severityList if $common_severity_reverse;
477 #print STDERR join(",",@severityList)."\n";
478
479 #        foreach my $severity(@debbugs::gSeverityList) {
480         foreach my $severity(@severityList) {
481             $severity = $debbugs::gDefaultSeverity if ($severity eq '');
482             next unless defined $section{${pending} . "_" . ${severity}};
483             $result .= "<HR><H2>$debbugs::gSeverityDisplay{$severity} - $displayshowpending{$pending}</H2>\n";
484             #$result .= "(A list of <a href=\"http://${debbugs::gWebDomain}/db/si/$pending$severity\">all such bugs</a> is available).\n";
485             #$result .= "(A list of all such bugs used to be available).\n";
486             $result .= "<UL>\n";
487             $result .= $section{$pending . "_" . $severity}; 
488             $result .= "</UL>\n";
489             $anydone = 1 if ($pending eq "done");
490          }
491     }
492
493     }
494     $result .= $debbugs::gHTMLExpireNote if $gRemoveAge and $anydone;
495     return $result;
496 }
497
498 sub countbugs {
499     my $bugfunc = shift;
500     if ($common_archive) {
501         open I, "<$debbugs::gSpoolDir/index.archive"
502             or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
503     } else {
504         open I, "<$debbugs::gSpoolDir/index.db"
505             or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
506     }
507
508     my %count = ();
509     while(<I>) 
510     {
511         if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
512             my @x = $bugfunc->(pkg => $1, bug => $2, status => $4, 
513                                submitter => $5, severity => $6, tags => $7);
514             local $_;
515             $count{$_}++ foreach @x;
516         }
517     }
518     close I;
519     return %count;
520 }
521
522 sub getbugs {
523     my $bugfunc = shift;
524     my $opt = shift;
525
526     my @result = ();
527
528     if (!$common_archive && defined $opt && 
529         -e "$debbugs::gSpoolDir/by-$opt.idx") 
530     {
531         my %lookup;
532 print STDERR "optimized\n" if ($debug);
533         tie %lookup, DB_File => "$debbugs::gSpoolDir/by-$opt.idx", O_RDONLY
534             or die "$0: can't open $debbugs::gSpoolDir/by-$opt.idx ($!)\n";
535         while ($key = shift) {
536             my $bugs = $lookup{$key};
537             if (defined $bugs) {
538                 push @result, (unpack 'N*', $bugs);
539             }
540         }
541         untie %lookup;
542 print STDERR "done optimized\n" if ($debug);
543     } else {
544         if ( $common_archive ) {
545             open I, "<$debbugs::gSpoolDir/index.archive" 
546                 or &quitcgi("$debbugs::gSpoolDir/index.archive: $!");
547         } else {
548             open I, "<$debbugs::gSpoolDir/index.db" 
549                 or &quitcgi("$debbugs::gSpoolDir/index.db: $!");
550         }
551         while(<I>) {
552             if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
553                 if ($bugfunc->(pkg => $1, bug => $2, status => $4,
554                             submitter => $5, severity => $6, tags => $7)) 
555                 {
556                     push (@result, $2);
557                 }
558             }
559         }
560         close I;
561     }
562     @result = sort {$a <=> $b} @result;
563     return \@result;
564 }
565
566 sub emailfromrfc822 {
567     my $email = shift;
568     $email =~ s/\s*\(.*\)\s*//;
569     $email = $1 if ($email =~ m/<(.*)>/);
570     return $email;
571 }
572
573 sub maintencoded {
574     my $input = shift;
575     my $encoded = '';
576
577     while ($input =~ m/\W/) {
578         $encoded.=$`.sprintf("-%02x_",unpack("C",$&));
579         $input= $';
580     }
581
582     $encoded.= $input;
583     $encoded =~ s/-2e_/\./g;
584     $encoded =~ s/^([^,]+)-20_-3c_(.*)-40_(.*)-3e_/$1,$2,$3,/;
585     $encoded =~ s/^(.*)-40_(.*)-20_-28_([^,]+)-29_$/,$1,$2,$3/;
586     $encoded =~ s/-20_/_/g;
587     $encoded =~ s/-([^_]+)_-/-$1/g;
588     return $encoded;
589 }
590
591 my $_maintainer;
592 sub getmaintainers {
593     return $_maintainer if $_maintainer;
594     my %maintainer;
595
596     open(MM,"$gMaintainerFile") or &quitcgi("open $gMaintainerFile: $!");
597     while(<MM>) {
598         next unless m/^(\S+)\s+(\S.*\S)\s*$/;
599         ($a,$b)=($1,$2);
600         $a =~ y/A-Z/a-z/;
601         $maintainer{$a}= $b;
602     }
603     close(MM);
604     if (defined $gMaintainerFileOverride) {
605         open(MM,"$gMaintainerFileOverride") or &quitcgi("open $gMaintainerFileOverride: $!");
606         while(<MM>) {
607             next unless m/^(\S+)\s+(\S.*\S)\s*$/;
608             ($a,$b)=($1,$2);
609             $a =~ y/A-Z/a-z/;
610             $maintainer{$a}= $b;
611         }
612         close(MM);
613     }
614     $_maintainer = \%maintainer;
615     return $_maintainer;
616 }
617
618 my $_pkgsrc;
619 my $_pkgcomponent;
620 sub getpkgsrc {
621     return $_pkgsrc if $_pkgsrc;
622     return {} unless defined $gPackageSource;
623     my %pkgsrc;
624     my %pkgcomponent;
625
626     open(MM,"$gPackageSource") or &quitcgi("open $gPackageSource: $!");
627     while(<MM>) {
628         next unless m/^(\S+)\s+(\S+)\s+(\S.*\S)\s*$/;
629         ($a,$b,$c)=($1,$2,$3);
630         $a =~ y/A-Z/a-z/;
631         $pkgsrc{$a}= $c;
632         $pkgcomponent{$a}= $b;
633     }
634     close(MM);
635     $_pkgsrc = \%pkgsrc;
636     $_pkgcomponent = \%pkgcomponent;
637     return $_pkgsrc;
638 }
639
640 sub getpkgcomponent {
641     return $_pkgcomponent if $_pkgcomponent;
642     getpkgsrc();
643     return $_pkgcomponent;
644 }
645
646 my $_pseudodesc;
647 sub getpseudodesc {
648     return $_pseudodesc if $_pseudodesc;
649     my %pseudodesc;
650
651     open(PSEUDO, "< $gPseudoDescFile") or &quitcgi("open $gPseudoDescFile: $!");
652     while(<PSEUDO>) {
653         next unless m/^(\S+)\s+(\S.*\S)\s*$/;
654         $pseudodesc{lc $1} = $2;
655     }
656     close(PSEUDO);
657     $_pseudodesc = \%pseudodesc;
658     return $_pseudodesc;
659 }
660
661 sub getbugstatus {
662     my $bugnum = shift;
663
664     my %status;
665
666     my $location = getbuglocation( $bugnum, 'summary' );
667     return {} if ( !$location );
668     %status = %{ readbug( $bugnum, $location ) };
669
670     $status{found_versions} = [];
671     $status{fixed_versions} = [];
672     if (defined $gVersionBugsDir and
673             (defined $common_version or defined $common_dist)) {
674         my $bughash = get_hashname($bugnum);
675         if (open BUGVER, "< $gVersionBugsDir/$bughash/$bugnum.versions") {
676             local $_;
677             while (<BUGVER>) {
678                 if (/^Found-in: (.*)/i) {
679                     $status{found_versions} = [split ' ', $1];
680                 } elsif (/^Fixed-in: (.*)/i) {
681                     $status{fixed_versions} = [split ' ', $1];
682                 }
683             }
684             close BUGVER;
685         }
686     }
687
688     $status{tags} = $status{keywords};
689     my %tags = map { $_ => 1 } split ' ', $status{tags};
690
691     $status{"package"} =~ s/\s*$//;
692     $status{"package"} = 'unknown' if ($status{"package"} eq '');
693     $status{"severity"} = 'normal' if ($status{"severity"} eq '');
694
695     $status{"pending"} = 'pending';
696     $status{"pending"} = 'forwarded'        if (length($status{"forwarded"}));
697     $status{"pending"} = 'pending-fixed'    if ($tags{pending});
698     $status{"pending"} = 'fixed'            if ($tags{fixed});
699
700     my $version;
701     if (defined $common_version) {
702         $version = $common_version;
703     } elsif (defined $common_dist) {
704         $version = getversion($status{package}, $common_dist, $common_arch);
705     }
706
707     if (defined $version) {
708         my $buggy = buggyversion($bugnum, $version, \%status);
709         if ($buggy eq 'absent') {
710             $status{"pending"} = 'absent';
711         } elsif ($buggy eq 'fixed') {
712             $status{"pending"} = 'done';
713         }
714     }
715     
716     if (length($status{done}) and
717             (not defined $version or not @{$status{fixed_versions}})) {
718         $status{"pending"} = 'done';
719     }
720
721     return \%status;
722 }
723
724 sub getsrcpkgs {
725     my $src = shift;
726     return () if !$src;
727     my %pkgsrc = %{getpkgsrc()};
728     my @pkgs;
729     foreach ( keys %pkgsrc ) {
730         push @pkgs, $_ if $pkgsrc{$_} eq $src;
731     }
732     return @pkgs;
733 }
734    
735 sub buglog {
736     my $bugnum = shift;
737     my $location = getbuglocation($bugnum, 'log');
738     return undef unless defined $location;
739     return getbugcomponent($bugnum, 'log', $location);
740 }
741
742 my %_versionobj;
743 sub buggyversion {
744     my ($bug, $ver, $status) = @_;
745     return '' unless defined $gVersionPackagesDir;
746     my $src = getpkgsrc()->{$status->{package}};
747     $src = $status->{package} unless defined $src;
748
749     my $tree;
750     if (exists $_versionobj{$src}) {
751         $tree = $_versionobj{$src};
752     } else {
753         $tree = Debbugs::Versions->new(\&DpkgVer::vercmp);
754         if (open VERFILE, "< $gVersionPackagesDir/$src") {
755             $tree->load(\*VERFILE);
756             close VERFILE;
757         }
758         $_versionobj{$src} = $tree;
759     }
760
761     return $tree->buggy($ver, $status->{found_versions},
762                         $status->{fixed_versions});
763 }
764
765 my %_versions;
766 sub getversion {
767     my ($pkg, $dist, $arch) = @_;
768     return undef unless defined $gVersionIndex;
769     $dist = 'unstable' unless defined $dist;
770     $arch = 'i386' unless defined $arch;
771
772     unless (tied %_versions) {
773         tie %_versions, 'MLDBM', $gVersionIndex, O_RDONLY
774             or die "can't open versions index: $!";
775     }
776
777     return $_versions{$pkg}{$dist}{$arch};
778 }
779
780 1;