]> git.donarmstrong.com Git - debbugs.git/blob - cgi/common.pl
[project @ 2000-11-17 11:46:07 by ajt]
[debbugs.git] / cgi / common.pl
1 #!/usr/bin/perl -w
2
3 my $common_archive = 0;
4 my $common_repeatmerged = 1;
5 my %common_include = ();
6 my %common_exclude = ();
7
8 sub set_option {
9     my ($opt, $val) = @_;
10     if ($opt eq "archive") { $common_archive = $val; }
11     if ($opt eq "repeatmerged") { $common_repeatmerged = $val; }
12     if ($opt eq "exclude") { %common_exclude = %{$val}; }
13     if ($opt eq "include") { %common_include = %{$val}; }
14 }
15
16 sub readparse {
17     my ($in, $key, $val, %ret);
18     if (defined $ENV{"QUERY_STRING"} && $ENV{"QUERY_STRING"} ne "") {
19         $in=$ENV{QUERY_STRING};
20     } elsif(defined $ENV{"REQUEST_METHOD"}
21         && $ENV{"REQUEST_METHOD"} eq "POST")
22     {
23         read(STDIN,$in,$ENV{CONTENT_LENGTH});
24     } else {
25         return;
26     }
27     foreach (split(/&/,$in)) {
28         s/\+/ /g;
29         ($key, $val) = split(/=/,$_,2);
30         $key=~s/%(..)/pack("c",hex($1))/ge;
31         $val=~s/%(..)/pack("c",hex($1))/ge;
32         $ret{$key}=$val;
33     }
34     return %ret;
35 }
36
37 sub quit {
38     my $msg = shift;
39     print "Content-Type: text/html\n\n";
40     print "<HTML><HEAD><TITLE>Error</TITLE></HEAD><BODY>\n";
41     print "An error occurred. Dammit.\n";
42     print "Error was: $msg.\n";
43     print "</BODY></HTML>\n";
44     exit 0;
45 }
46
47 #sub abort {
48 #    my $msg = shift;
49 #    my $Archive = $common_archive ? "archive" : "";
50 #    print header . start_html("Sorry");
51 #    print "Sorry bug #$msg doesn't seem to be in the $Archive database.\n";
52 #    print end_html;
53 #    exit 0;
54 #}
55
56 sub htmlindexentry {
57     my $ref = shift;
58     my %status = getbugstatus($ref);
59     return htmlindexentrystatus(%status) if (%status);
60     return "";
61 }
62
63 sub htmlindexentrystatus {
64     my $s = shift;
65     my %status = %{$s};
66
67     my $result = "";
68
69     if  ($status{severity} eq 'normal') {
70         $showseverity = '';
71     } elsif (grep($status{severity} eq $_, @debbugs::gStrongSeverities)) {
72         $showseverity = "<strong>Severity: $status{severity}</strong>;\n";
73     } else {
74         $showseverity = "Severity: <em>$status{severity}</em>;\n";
75     }
76
77     $result .= "Package: <a href=\"" . pkgurl($status{"package"}) . "\">"
78                . "<strong>" . htmlsanit($status{"package"}) . "</strong></a>;\n"
79                if (length($status{"package"}));
80     $result .= $showseverity;
81     $result .= "Reported by: " . htmlsanit($status{originator});
82     $result .= ";\nTags: <strong>" 
83                  . htmlsanit(join(", ", sort(split(/\s+/, $status{tags}))))
84                  . "</strong>"
85                        if (length($status{tags}));
86
87     my @merged= split(/ /,$status{mergedwith});
88     my $mseparator= ";\nmerged with ";
89     for my $m (@merged) {
90         $result .= $mseparator."<A href=\"" . bugurl($m) . "\">#$m</A>";
91         $mseparator= ", ";
92     }
93
94     if (length($status{done})) {
95         $result .= ";\n<strong>Done:</strong> " . htmlsanit($status{done});
96     } elsif (length($status{forwarded})) {
97         $result .= ";\n<strong>Forwarded</strong> to "
98                    . htmlsanit($status{forwarded});
99     } else {
100         my $daysold = int((time - $status{date}) / 86400);   # seconds to days
101         if ($daysold >= 7) {
102             my $font = "";
103             my $efont = "";
104             $font = "em" if ($daysold > 30);
105             $font = "strong" if ($daysold > 60);
106             $efont = "</$font>" if ($font);
107             $font = "<$font>" if ($font);
108
109             my $yearsold = int($daysold / 364);
110             $daysold = $daysold - $yearsold * 364;
111
112             $result .= ";\n $font";
113             $result .= "1 year and " if ($yearsold == 1);
114             $result .= "$yearsold years and " if ($yearsold > 1);
115             $result .= "1 day old" if ($daysold == 1);
116             $result .= "$daysold days old" if ($daysold != 1);
117             $result .= "$efont";
118         }
119     }
120
121     $result .= ".";
122
123     return $result;
124 }
125
126 sub submitterurl {
127     my $ref = shift || "";
128     my $params = "submitter=" . emailfromrfc822($ref);
129     $params .= "&archive=yes" if ($common_archive);
130     $params .= "&repeatmerged=yes" if ($common_repeatmerged);
131     return $debbugs::gCGIDomain . "pkgreport.cgi" . "?" . $params;
132 }
133
134 sub mainturl {
135     my $ref = shift || "";
136     my $params = "maint=" . emailfromrfc822($ref);
137     $params .= "&archive=yes" if ($common_archive);
138     $params .= "&repeatmerged=yes" if ($common_repeatmerged);
139     return $debbugs::gCGIDomain . "pkgreport.cgi" . "?" . $params;
140 }
141
142 sub pkgurl {
143     my $ref = shift;
144     my $params = "pkg=$ref";
145     $params .= "&archive=yes" if ($common_archive);
146     $params .= "&repeatmerged=yes" if ($common_repeatmerged);
147     
148     return $debbugs::gCGIDomain . "pkgreport.cgi" . "?" . "$params";
149 }
150
151 sub htmlsanit {
152     my %saniarray = ('<','lt', '>','gt', '&','amp', '"','quot');
153     my $in = shift || "";
154     my $out;
155     while ($in =~ m/[<>&"]/) {
156         $out .= $`. '&'. $saniarray{$&}. ';';
157         $in = $';
158     }
159     $out .= $in;
160     return $out;
161 }
162
163 sub bugurl {
164     my $ref = shift;
165     my $params = "bug=$ref";
166     foreach my $val (@_) {
167         $params .= "\&msg=$1" if ($val =~ /^msg=([0-9]+)/);
168         $params .= "\&archive=yes" if (!$common_archive && $val =~ /^archive.*$/);
169     }
170     $params .= "&archive=yes" if ($common_archive);
171     $params .= "&repeatmerged=yes" if ($common_repeatmerged);
172
173     return $debbugs::gCGIDomain . "bugreport.cgi" . "?" . "$params";
174 }
175
176 sub packageurl {
177     my $ref = shift;
178     return $debbugs::gCGIDomain . "package.cgi" . "?" . "package=$ref";
179 }
180
181 sub allbugs {
182     my @bugs = ();
183
184     opendir(D, "$debbugs::gSpoolDir/db") or &quit("opendir db: $!");
185     @bugs = sort {$a<=>$b} grep s/\.status$//,
186                  (grep m/^[0-9]+\.status$/,
187                  (readdir(D)));
188     closedir(D);
189
190     return @bugs;
191 }
192
193 sub htmlizebugs {
194     my @bugs = @_;
195
196     my %section = ();
197
198     my %displayshowpending = ("pending", "outstanding",
199                               "done", "resolved",
200                               "forwarded", "forwarded to upstream software authors");
201
202     if (@bugs == 0) {
203         return "<HR><H2>No reports found!</H2></HR>\n";
204     }
205
206     foreach my $bug (sort {$a<=>$b} @bugs) {
207         my %status = getbugstatus($bug);
208         next unless %status;
209         my @merged = sort {$a<=>$b} ($bug, split(/ /, $status{mergedwith}));
210         next unless ($common_repeatmerged || $bug == $merged[0]);
211         if (%common_include) {
212             my $okay = 0;
213             foreach my $t (split /\s+/, $status{tags}) {
214                 $okay = 1, last if (defined $common_include{$t});
215             }
216             next unless ($okay);
217         }
218         if (%common_exclude) {
219             my $okay = 1;
220             foreach my $t (split /\s+/, $status{tags}) {
221                 $okay = 0, last if (defined $comon_exclude{$t});
222             }
223             next unless ($okay);
224         }
225             
226         $section{$status{pending} . "_" . $status{severity}} .=
227             sprintf "<li><a href=\"%s\">#%d: %s</a>\n<br>",
228                 bugurl($bug), $bug, htmlsanit($status{subject});
229         $section{$status{pending} . "_" . $status{severity}} .=
230             htmlindexentrystatus(\%status) . "\n";
231     }
232
233     my $result = "";
234     my $anydone = 0;
235     foreach my $pending (qw(pending forwarded done)) {
236         foreach my $severity(@debbugs::gSeverityList) {
237             $severity = $debbugs::gDefaultSeverity if ($severity eq '');
238             next unless defined $section{${pending} . "_" . ${severity}};
239             $result .= "<HR><H2>$debbugs::gSeverityDisplay{$severity} - $displayshowpending{$pending}</H2>\n";
240             $result .= "(A list of <a href=\"http://www.debian.org/Bugs/db/si/$pending$severity\">all such bugs</a> is available).\n";
241             $result .= "<UL>\n";
242             $result .= $section{$pending . "_" . $severity}; 
243             $result .= "</UL>\n";
244             $anydone = 1 if ($pending eq "done");
245          }
246     }
247
248     $result .= $debbugs::gHTMLExpireNote if ($anydone);
249     return $result;
250 }
251
252 sub submitterbugs {
253     my $submitter = shift;
254     my $chk = sub {
255         my %d = @_;
256         ($subemail = $d{"submitter"}) =~ s/\s*\(.*\)\s*//;
257         if ($subemail =~ m/<(.*)>/) { $subemail = $1 }
258         return $subemail eq $submitter;
259     };
260     return getbugs($chk);
261 }
262
263 sub severitybugs {
264     my $status = shift;
265     my $severity = shift;
266     my $chk = sub {
267         my %d = @_; 
268         return ($d{"severity"} eq $severity) && ($d{"status"} eq $status); 
269     };
270     return getbugs($chk);
271 }
272
273 sub maintbugs {
274     my $maint = shift;
275     my %maintainers = getmaintainers();
276     my $chk = sub {
277         my %d = @_;
278         ($maintemail = $maintainers{$d{"pkg"}} || "") =~ s/\s*\(.*\)\s*//;
279         if ($maintemail =~ m/<(.*)>/) { $maintemail = $1 }
280         return $maintemail eq $maint;
281     };
282     return getbugs($chk);
283 }
284
285 sub maintencbugs {
286     my $maintenc = shift;
287     my %maintainers = getmaintainers();
288     return getbugs(sub {my %d=@_; return maintencoded($maintainers{$d{"pkg"}} || "") eq $maintenc});
289 }
290
291 sub pkgbugs {
292     my $inpkg = shift;
293     return getbugs( sub { my %d = @_; return $inpkg eq $d{"pkg"} });
294 }
295
296 sub countbugs {
297     my $bugfunc = shift;
298     if ($common_archive) {
299         open I, "<$debbugs::gSpoolDir/index.archive" or &quit("bugindex: $!");
300     } else {
301         open I, "<$debbugs::gSpoolDir/index.db" or &quit("bugindex: $!");
302     }
303
304     my %count = ();
305     while(<I>) 
306     {
307         if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
308             my $x = $bugfunc->(pkg => $1, bug => $2, status => $4, 
309                                submitter => $5, severity => $6, tags => $7);
310             $count{$x}++;
311         }
312     }
313     close I;
314     return %count;
315 }
316
317 sub getbugs {
318     my $bugfunc = shift;
319
320     if ( $common_archive ) {
321         open I, "<$debbugs::gSpoolDir/index.archive" or &quit("bugindex: $!");
322     } else {
323         open I, "<$debbugs::gSpoolDir/index.db" or &quit("bugindex: $!");
324     }
325     
326     my @result = ();
327     while(<I>) 
328     {
329         if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) {
330             if ($bugfunc->(pkg => $1, bug => $2, status => $4, submitter => $5,
331                            severity => $6, tags => $7))
332             {
333                 push (@result, $2);
334                 #last if (@result > 400);
335             }
336         }
337     }
338     close I;
339     return sort {$a <=> $b} @result;
340 }
341
342 sub pkgbugsindex {
343     my %descstr = ();
344     if ( $common_archive ) {
345         open I, "<$debbugs::gSpoolDir/index.archive" or &quit("bugindex: $!");
346     } else {
347         open I, "<$debbugs::gSpoolDir/index.db" or &quit("bugindex: $!");
348     }
349     while(<I>) { 
350         $descstr{ $1 } = 1 if (m/^(\S+)/);
351     }
352     close(I);
353     return %descstr;
354 }
355
356 sub emailfromrfc822 {
357     my $email = shift;
358     $email =~ s/\s*\(.*\)\s*//;
359     $email = $1 if ($email =~ m/<(.*)>/);
360     return $email;
361 }
362
363 sub maintencoded {
364     my $input = shift;
365     my $encoded = '';
366
367     while ($input =~ m/\W/) {
368         $encoded.=$`.sprintf("-%02x_",unpack("C",$&));
369         $input= $';
370     }
371
372     $encoded.= $input;
373     $encoded =~ s/-2e_/\./g;
374     $encoded =~ s/^([^,]+)-20_-3c_(.*)-40_(.*)-3e_/$1,$2,$3,/;
375     $encoded =~ s/^(.*)-40_(.*)-20_-28_([^,]+)-29_$/,$1,$2,$3/;
376     $encoded =~ s/-20_/_/g;
377     $encoded =~ s/-([^_]+)_-/-$1/g;
378     return $encoded;
379 }
380
381 sub getmaintainers {
382     my %maintainer;
383
384     open(MM,"$gMaintainerFile") or &quit("open $gMaintainerFile: $!");
385     while(<MM>) {
386         next unless m/^(\S+)\s+(\S.*\S)\s*$/;
387         ($a,$b)=($1,$2);
388         $a =~ y/A-Z/a-z/;
389         $maintainer{$a}= $b;
390     }
391     close(MM);
392
393     return %maintainer;
394 }
395
396 sub getbugstatus {
397     my $bugnum = shift;
398
399     my %status;
400
401     unless (open(S,"$gSpoolDir/db/$bugnum.status")) {
402         my $archdir = sprintf "%02d", $bugnum % 100;
403         open(S,"$gSpoolDir/archive/$archdir/$bugnum.status" ) or return ();
404     }
405     my @lines = qw(originator date subject msgid package tags done
406                         forwarded mergedwith severity);
407     while(<S>) {
408         chomp;
409         $status{shift @lines} = $_;
410     }
411     close(S);
412     $status{shift @lines} = '' while(@lines);
413
414     $status{"package"} =~ s/\s*$//;
415     $status{"package"} = 'unknown' if ($status{"package"} eq '');
416     $status{"severity"} = 'normal' if ($status{"severity"} eq '');
417
418     $status{"pending"} = 'pending';
419     $status{"pending"} = 'forwarded' if (length($status{"forwarded"}));
420     $status{"pending"} = 'done'      if (length($status{"done"}));
421
422     return %status;
423 }
424
425 sub buglog {
426     my $bugnum = shift;
427     my $res;
428
429     $res = "$gSpoolDir/db/$bugnum.log"; 
430     return $res if ( -e $res );
431
432     my $archdir = sprintf "%02d", $bugnum % 100;
433     $res = "$gSpoolDir/archive/$archdir/$bugnum.log";
434     return $res if ( -e $res );
435
436     return "";
437 }
438
439 1