]> git.donarmstrong.com Git - debbugs.git/blob - cgi/pkgreport.cgi
[project @ 2005-10-06 03:46:13 by ajt]
[debbugs.git] / cgi / pkgreport.cgi
1 #!/usr/bin/perl -wT
2
3 package debbugs;
4
5 use strict;
6 use POSIX qw(strftime tzset nice);
7
8 #require '/usr/lib/debbugs/errorlib';
9 require './common.pl';
10
11 require '/etc/debbugs/config';
12 require '/etc/debbugs/text';
13
14 use Debbugs::User;
15 my $cats = 5;
16
17 use vars qw($gPackagePages $gWebDomain);
18
19 if (defined $ENV{REQUEST_METHOD} and $ENV{REQUEST_METHOD} eq 'HEAD') {
20     print "Content-Type: text/html; charset=utf-8\n\n";
21     exit 0;
22 }
23
24 nice(5);
25
26 my $userAgent = detect_user_agent();
27 my %param = readparse();
28
29 my $repeatmerged = ($param{'repeatmerged'} || "yes") eq "yes";
30 my $archive = ($param{'archive'} || "no") eq "yes";
31 my $include = $param{'&include'} || $param{'include'} || "";
32 my $exclude = $param{'&exclude'} || $param{'exclude'} || "";
33
34 my $users = $param{'users'} || "";
35
36 my $ordering = $param{'ordering'};
37 my $raw_sort = ($param{'raw'} || "no") eq "yes";
38 my $old_view = ($param{'oldview'} || "no") eq "yes";
39 unless (defined $ordering) {
40    $ordering = "normal";
41    $ordering = "old" if $old_view;
42    $ordering = "raw" if $raw_sort;
43 }
44
45 my $bug_rev = ($param{'bug-rev'} || "no") eq "yes";
46 my $pend_rev = ($param{'pend-rev'} || "no") eq "yes";
47 my $sev_rev = ($param{'sev-rev'} || "no") eq "yes";
48 my $pend_exc = $param{'&pend-exc'} || $param{'pend-exc'} || "";
49 my $pend_inc = $param{'&pend-inc'} || $param{'pend-inc'} || "";
50 my $sev_exc = $param{'&sev-exc'} || $param{'sev-exc'} || "";
51 my $sev_inc = $param{'&sev-inc'} || $param{'sev-inc'} || "";
52 my $maxdays = ($param{'maxdays'} || -1);
53 my $mindays = ($param{'mindays'} || 0);
54 my $version = $param{'version'} || undef;
55 my $dist = $param{'dist'} || undef;
56 my $arch = $param{'arch'} || undef;
57 my $show_list_header = ($param{'show_list_header'} || $userAgent->{'show_list_header'} || "yes" ) eq "yes";
58 my $show_list_footer = ($param{'show_list_footer'} || $userAgent->{'show_list_footer'} || "yes" ) eq "yes";
59
60 my @p = (
61   "pending:pending,forwarded,pending-fixed,fixed,done,absent",
62   "severity:critical,grave,serious,important,normal,minor,wishlist",
63   "pending=pending+tag=wontfix,pending=pending+tag=moreinfo,pending=pending+tag=patch,pending=pending+tag=confirmed,pending=pending");
64 my @t = (
65   "Outstanding,Forwarded,Pending Upload,Fixed in NMU,Resolved,From other Branch,Unknown Pending Status",
66   "Critical,Grave,Serious,Important,Normal,Minor,Wishlist,Unknown Severity",
67   "Will Not Fix,More information needed,Patch Available,Confirmed,Unclassified");
68 my @o = ("0,1,2,3,4,5,6","0,1,2,3,4,5,6,7","2,3,4,1,0,5");
69 my @n = ("Status", "Severity", "Classification");
70
71 if ($ordering eq "old") {
72     splice @p, 2, 1;
73     splice @t, 2, 1;
74     splice @o, 2, 1;
75     splice @n, 2, 1;
76 }
77 $o[0] = scalar reverse($o[0]) if ($pend_rev);
78 $o[1] = scalar reverse($o[1]) if ($sev_rev);
79
80 if (!defined $param{"pri0"} && $ordering =~ m/^user(\d+)$/) {
81     my $id = $1;
82     my $l = 0;
83     if (defined $param{"cat${id}_users"}) {
84         $users .= "," . $param{"cat${id}_users"};
85     }
86     while (defined $param{"cat${id}_nam$l"}) {
87         my ($n, $p, $t, $o) =
88             map { $param{"cat${id}_${_}$l"} || "" }
89                 ("nam", "pri", "ttl", "ord");
90         if ($p eq "") {
91             if ($n eq "status") {
92                 ($p, $t, $o) = ($p[0], $t[0], $o[0]);
93             } elsif ($n eq "severity") {
94                 ($p, $t, $o) = ($p[1], $t[1], $o[1])
95             } else {
96                 $ordering = "raw";
97                 last;
98             }
99         }
100         $param{"nam$l"} = $n;
101         $param{"pri$l"} = $p;
102         $param{"ttl$l"} = $t;
103         $param{"ord$l"} = $o;
104         $l++;
105     }
106 }
107 if (defined $param{"pri0"}) {
108     my $i = 0;
109     @p = (); @o = (); @t = (); @n = ();
110     while (defined $param{"pri$i"}) {
111         push @p, $param{"pri$i"};
112         push @o, $param{"ord$i"} || "";
113         push @t, $param{"ttl$i"} || "";
114         push @n, $param{"nam$i"} || "";
115         $i++;
116     }
117 }
118 for my $x (@p) {
119     next if "$x," =~ m/^(pending|severity|tag):(([*]|[a-z0-9.-]+),)+$/;
120     next if "$x," =~ m/^((pending|severity|tag)=([*]|[a-z0-9.-]+)[,+])+/;
121     quitcgi("Bad syntax in Priority: $x");
122 }
123
124 my @names; my @prior; my @title; my @order;
125 for my $i (0..$#p) {
126     push @prior, [ make_order_list($p[$i]) ];
127     if ($n[$i]) {
128         push @names, $n[$i];
129     } elsif ($p[$i] =~ m/^([^:]+):/) {
130         push @names, $1;
131     } else {
132         push @names, "Bug attribute #" . (1+$i);
133     }
134     if ($o[$i]) {
135         push @order, [ split /,/, $o[$i] ];
136     } else {
137         push @order, [ 0..$#{$prior[$i]} ];
138     }
139     my @t = split /,/, $t[$i];
140     push @t, map { toenglish($prior[$i]->[$_]) } ($#t+1)..($#{$prior[$i]});
141     push @title, [@t];
142 }
143
144 sub toenglish {
145     my $expr = shift;
146     $expr =~ s/[+]/ and /g;
147     $expr =~ s/[a-z]+=//g;
148     return $expr;
149 }
150
151 {
152     if (defined $param{'vt'}) {
153         my $vt = $param{'vt'};
154         if ($vt eq "none") { $dist = undef; $arch = undef; $version = undef; }
155         if ($vt eq "bysuite") {
156             $version = undef;
157             $arch = undef if ($arch eq "any");
158         }
159         if ($vt eq "bypkg" || $vt eq "bysrc") { $dist = undef; $arch = undef; }
160     }
161     if (defined $param{'includesubj'}) {
162         my $is = $param{'includesubj'};
163         $include .= "," . join(",", map { "subj:$_" } (split /[\s,]+/, $is));
164     }
165     if (defined $param{'excludesubj'}) {
166         my $es = $param{'excludesubj'};
167         $exclude .= "," . join(",", map { "subj:$_" } (split /[\s,]+/, $es));
168     }
169 }
170
171
172 my %bugusertags;
173 my %ut;
174 for my $user (split /[\s*,]+/, $users) {
175     next unless ($user =~ m/..../);
176     Debbugs::User::read_usertags(\%ut, $user);
177 }
178
179 my ($pkg, $src, $maint, $maintenc, $submitter, $severity, $status, $tag, $usertag);
180
181 my %which = (
182         'pkg' => \$pkg,
183         'src' => \$src,
184         'maint' => \$maint,
185         'maintenc' => \$maintenc,
186         'submitter' => \$submitter,
187         'severity' => \$severity,
188         'tag' => \$tag,
189         'usertag' => \$usertag,
190         );
191 my @allowedEmpty = ( 'maint' );
192
193 my $found;
194 foreach ( keys %which ) {
195         $status = $param{'status'} || 'open' if /^severity$/;
196         if (($found = $param{$_})) {
197                 ${ $which{$_} } = $found;
198                 last;
199         }
200 }
201 if (!$found && !$archive) {
202         foreach ( @allowedEmpty ) {
203                 if (exists($param{$_})) {
204                         ${ $which{$_} } = '';
205                         $found = 1;
206                         last;
207                 }
208         }
209 }
210 if (!$found) {
211         my $which;
212         if (($which = $param{'which'})) {
213                 if (grep( /^\Q$which\E$/, @allowedEmpty)) {
214                         ${ $which{$which} } = $param{'data'};
215                         $found = 1;
216                 } elsif (($found = $param{'data'})) {
217                         ${ $which{$which} } = $found if (exists($which{$which}));
218                 }
219         }
220 }
221 quitcgi("You have to choose something to select by") if (!$found);
222
223 if (defined $usertag) {
224     my %select_ut = ();
225     my ($u, $t) = split /:/, $usertag, 2;
226     Debbugs::User::read_usertags(\%select_ut, $u);
227     unless (defined $t && $t ne "") {
228         $t = join(",", keys(%select_ut));
229     }
230
231     Debbugs::User::read_usertags(\%ut, $u);
232     $tag = $t;
233 }
234
235 for my $t (keys %ut) {
236     for my $b (@{$ut{$t}}) {
237         $bugusertags{$b} = [] unless defined $bugusertags{$b};
238         push @{$bugusertags{$b}}, $t;
239     }
240 }
241
242 my $Archived = $archive ? " Archived" : "";
243
244 my $this = "";
245
246 my %indexentry;
247 my %strings = ();
248
249 $ENV{"TZ"} = 'UTC';
250 tzset();
251
252 my $dtime = strftime "%a, %e %b %Y %T UTC", localtime;
253 my $tail_html = $debbugs::gHTMLTail;
254 $tail_html = $debbugs::gHTMLTail;
255 $tail_html =~ s/SUBSTITUTE_DTIME/$dtime/;
256
257 set_option("repeatmerged", $repeatmerged);
258 set_option("archive", $archive);
259 set_option("include", $include);
260 set_option("exclude", $exclude);
261 set_option("pend-exc", $pend_exc);
262 set_option("pend-inc", $pend_inc);
263 set_option("sev-exc", $sev_exc);
264 set_option("sev-inc", $sev_inc);
265 set_option("maxdays", $maxdays);
266 set_option("mindays", $mindays);
267 set_option("version", $version);
268 set_option("dist", $dist);
269 set_option("arch", $arch);
270 set_option("use-bug-idx", defined($param{'use-bug-idx'}) ? $param{'use-bug-idx'} : 0);
271 set_option("show_list_header", $show_list_header);
272 set_option("show_list_footer", $show_list_footer);
273 set_option("bugusertags", \%bugusertags);
274
275 my $title;
276 my @bugs;
277 if (defined $pkg) {
278   $title = "package $pkg";
279   if (defined $version) {
280     $title .= " (version $version)";
281   } elsif (defined $dist) {
282     $title .= " in $dist";
283     my $verdesc = getversiondesc($pkg);
284     $title .= " ($verdesc)" if defined $verdesc;
285   }
286   my @pkgs = split /,/, $pkg;
287   @bugs = @{getbugs(sub {my %d=@_;
288                          foreach my $try (splitpackages($d{"pkg"})) {
289                            return 1 if grep($try eq $_, @pkgs);
290                          }
291                          return 0;
292                         }, 'package', @pkgs)};
293 } elsif (defined $src) {
294   $title = "source $src";
295   set_option('arch', 'source');
296   if (defined $version) {
297     $title .= " (version $version)";
298   } elsif (defined $dist) {
299     $title .= " in $dist";
300     my $verdesc = getversiondesc($src);
301     $title .= " ($verdesc)" if defined $verdesc;
302   }
303   my @pkgs = ();
304   my @srcs = split /,/, $src;
305   foreach my $try (@srcs) {
306     push @pkgs, getsrcpkgs($try);
307     push @pkgs, $try if ( !grep(/^\Q$try\E$/, @pkgs) );
308   }
309   @bugs = @{getbugs(sub {my %d=@_;
310                          foreach my $try (splitpackages($d{"pkg"})) {
311                            return 1 if grep($try eq $_, @pkgs);
312                          }
313                          return 0;
314                         }, 'package', @pkgs)};
315 } elsif (defined $maint) {
316   my %maintainers = %{getmaintainers()};
317   $title = "maintainer $maint";
318   $title .= " in $dist" if defined $dist;
319   if ($maint eq "") {
320     @bugs = @{getbugs(sub {my %d=@_;
321                            foreach my $try (splitpackages($d{"pkg"})) {
322                              return 1 if !getparsedaddrs($maintainers{$try});
323                            }
324                            return 0;
325                           })};
326   } else {
327     my @maints = split /,/, $maint;
328     my @pkgs = ();
329     foreach my $try (@maints) {
330       foreach my $p (keys %maintainers) {
331         my @me = getparsedaddrs($maintainers{$p});
332         push @pkgs, $p if grep { $_->address eq $try } @me;
333       }
334     }
335     @bugs = @{getbugs(sub {my %d=@_;
336                            foreach my $try (splitpackages($d{"pkg"})) {
337                              my @me = getparsedaddrs($maintainers{$try});
338                              return 1 if grep { $_->address eq $maint } @me;
339                            }
340                            return 0;
341                           }, 'package', @pkgs)};
342   }
343 } elsif (defined $maintenc) {
344   my %maintainers = %{getmaintainers()};
345   $title = "encoded maintainer $maintenc";
346   $title .= " in $dist" if defined $dist;
347   @bugs = @{getbugs(sub {my %d=@_; 
348                          foreach my $try (splitpackages($d{"pkg"})) {
349                            my @me = getparsedaddrs($maintainers{$try});
350                            return 1 if grep {
351                              maintencoded($_->address) eq $maintenc
352                            } @me;
353                          }
354                          return 0;
355                         })};
356 } elsif (defined $submitter) {
357   $title = "submitter $submitter";
358   $title .= " in $dist" if defined $dist;
359   my @submitters = split /,/, $submitter;
360   @bugs = @{getbugs(sub {my %d=@_;
361                          my @se = getparsedaddrs($d{"submitter"} || "");
362                          foreach my $try (@submitters) {
363                            return 1 if grep { $_->address eq $try } @se;
364                          }
365                         }, 'submitter-email', @submitters)};
366 } elsif (defined($severity) && defined($status)) {
367   $title = "$status $severity bugs";
368   $title .= " in $dist" if defined $dist;
369   my @severities = split /,/, $severity;
370   my @statuses = split /,/, $status;
371   @bugs = @{getbugs(sub {my %d=@_;
372                        return (grep($d{"severity"} eq $_, @severities))
373                          && (grep($d{"status"} eq $_, @statuses));
374                      })};
375 } elsif (defined($severity)) {
376   $title = "$severity bugs";
377   $title .= " in $dist" if defined $dist;
378   my @severities = split /,/, $severity;
379   @bugs = @{getbugs(sub {my %d=@_;
380                        return (grep($d{"severity"} eq $_, @severities));
381                      }, 'severity', @severities)};
382 } elsif (defined($tag)) {
383   $title = "bugs tagged $tag";
384   $title .= " in $dist" if defined $dist;
385   my @tags = split /,/, $tag;
386   my %bugs = ();
387   for my $t (@tags) {
388       for my $b (@{$ut{$t}}) {
389           $bugs{$b} = 1;
390        }
391   }
392   @bugs = @{getbugs(sub {my %d = @_;
393                          return 1 if $bugs{$d{"bug"}};
394                          my %tags = map { $_ => 1 } split ' ', $d{"tags"};
395                          return grep(exists $tags{$_}, @tags);
396                         })};
397 }
398
399 my $result = pkg_htmlizebugs(\@bugs);
400
401 print "Content-Type: text/html; charset=utf-8\n\n";
402
403 print "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">\n";
404 print "<HTML><HEAD>\n" . 
405     "<TITLE>$debbugs::gProject$Archived $debbugs::gBug report logs: $title</TITLE>\n" .
406     '<link rel="stylesheet" href="/css/bugs.css" type="text/css">' .
407     "</HEAD>\n" .
408     '<BODY onload="pagemain();">' .
409     "\n";
410 print "<H1>" . "$debbugs::gProject$Archived $debbugs::gBug report logs: $title" .
411       "</H1>\n";
412
413 my $showresult = 1;
414
415 if (defined $pkg || defined $src) {
416     my $showpkg = (defined $pkg) ? $pkg : "source package $src";
417     my %maintainers = %{getmaintainers()};
418     my $maint = $pkg ? $maintainers{$pkg} : $maintainers{$src} ? $maintainers{$src} : undef;
419     if (defined $maint) {
420         print '<p>';
421         print htmlmaintlinks(sub { $_[0] == 1 ? "Maintainer for $showpkg is "
422                                               : "Maintainers for $showpkg are "
423                                  },
424                              $maint);
425         print ".</p>\n";
426     } else {
427         print "<p>No maintainer for $showpkg. Please do not report new bugs against this package.</p>\n";
428     }
429     if (defined $maint or @bugs) {
430         my %pkgsrc = %{getpkgsrc()};
431         my $srcforpkg;
432         if (defined $pkg) {
433             $srcforpkg = $pkgsrc{$pkg};
434             defined $srcforpkg or $srcforpkg = $pkg;
435         }
436         my @pkgs = getsrcpkgs($pkg ? $srcforpkg : $src);
437         undef $srcforpkg unless @pkgs;
438         @pkgs = grep( !/^\Q$pkg\E$/, @pkgs ) if ( $pkg );
439         if ( @pkgs ) {
440             @pkgs = sort @pkgs;
441             if ($pkg) {
442                     print "<p>You may want to refer to the following packages that are part of the same source:\n";
443             } else {
444                     print "<p>You may want to refer to the following individual bug pages:\n";
445             }
446             push @pkgs, $src if ( $src && !grep(/^\Q$src\E$/, @pkgs) );
447             print join( ", ", map( "<A href=\"" . pkgurl($_) . "\">$_</A>", @pkgs ) );
448             print ".\n";
449         }
450         my @references;
451         my $pseudodesc = getpseudodesc();
452         if ($pkg and defined($pseudodesc) and exists($pseudodesc->{$pkg})) {
453             push @references, "to the <a href=\"http://${debbugs::gWebDomain}/pseudo-packages${debbugs::gHTMLSuffix}\">list of other pseudo-packages</a>";
454         } else {
455             if ($pkg and defined $debbugs::gPackagePages) {
456                 push @references, sprintf "to the <a href=\"%s\">%s package page</a>", urlsanit("http://${debbugs::gPackagePages}/$pkg"), htmlsanit("$pkg");
457             }
458             if (defined $debbugs::gSubscriptionDomain) {
459                 my $ptslink = $pkg ? $srcforpkg : $src;
460                 push @references, "to the <a href=\"http://$debbugs::gSubscriptionDomain/$ptslink\">Package Tracking System</a>";
461             }
462             # Only output this if the source listing is non-trivial.
463             if ($pkg and $srcforpkg and (@pkgs or $pkg ne $srcforpkg)) {
464                 push @references, sprintf "to the source package <a href=\"%s\">%s</a>'s bug page", srcurl($srcforpkg), htmlsanit($srcforpkg);
465             }
466         }
467         if ($pkg) {
468             set_option("archive", !$archive);
469             push @references, sprintf "to the <a href=\"%s\">%s reports for %s</a>", pkgurl($pkg), ($archive ? "active" : "archived"), htmlsanit($pkg);
470             set_option("archive", $archive);
471         }
472         if (@references) {
473             $references[$#references] = "or $references[$#references]" if @references > 1;
474             print "<p>You might like to refer ", join(", ", @references), ".</p>\n";
475         }
476         print "<p>If you find a bug not listed here, please\n";
477         printf "<a href=\"%s\">report it</a>.</p>\n",
478                urlsanit("http://${debbugs::gWebDomain}/Reporting${debbugs::gHTMLSuffix}");
479     } else {
480         print "<p>There is no record of the " .
481               (defined($pkg) ? htmlsanit($pkg) . " package"
482                              : htmlsanit($src) . " source package") .
483               ", and no bugs have been filed against it.</p>";
484         $showresult = 0;
485     }
486 } elsif (defined $maint || defined $maintenc) {
487     print "<p>Note that maintainers may use different Maintainer fields for\n";
488     print "different packages, so there may be other reports filed under\n";
489     print "different addresses.\n";
490 } elsif (defined $submitter) {
491     print "<p>Note that people may use different email accounts for\n";
492     print "different bugs, so there may be other reports filed under\n";
493     print "different addresses.\n";
494 }
495
496 print $result if $showresult;
497
498 print pkg_javascript() . "\n";
499 print "<h2 class=\"outstanding\"><a class=\"options\" href=\"javascript:toggle(1)\">Options</a></h2>\n";
500 print "<div id=\"a_1\">\n";
501 printf "<form action=\"%s\" method=POST>\n", myurl();
502
503 print "<table class=\"forms\">\n";
504
505 my ($checked_any, $checked_sui, $checked_ver) = ("", "", "");
506 if (defined $dist) {
507   $checked_sui = "CHECKED";
508 } elsif (defined $version) {
509   $checked_ver = "CHECKED";
510 } else {
511   $checked_any = "CHECKED";
512 }
513
514 print "<tr><td>Show bugs applicable to</td>\n";
515 print "    <td><input id=\"b_1_1\" name=vt value=none type=radio onchange=\"enable(1);\" $checked_any>anything</td></tr>\n";
516 print "<tr><td></td>";
517 print "    <td><input id=\"b_1_2\" name=vt value=bysuite type=radio onchange=\"enable(1);\" $checked_sui>" . pkg_htmlselectsuite(1,2,1) . " for " . pkg_htmlselectarch(1,2,2) . "</td></tr>\n";
518
519 if (defined $pkg) {
520     my $v = $version || "";
521     print "<tr><td></td>";
522     print "    <td><input id=\"b_1_3\" name=vt value=bypkg type=radio onchange=\"enable(1);\" $checked_ver>$pkg version <input id=\"b_1_3_1\" name=version value=\"$v\"></td></tr>\n";
523 } elsif (defined $src) {
524     my $v = $version || "";
525     print "<tr><td></td>";
526     print "    <td><input name=vt value=bysrc type=radio onchange=\"enable(1);\" $checked_ver>$src version <input id=\"b_1_3_1\" name=version value=\"$v\"></td></tr>\n";
527 }
528 print "<tr><td>&nbsp;</td></tr>\n";
529
530 my $includetags = join(" ", grep { !m/^subj:/i } split /[\s,]+/, $include);
531 my $excludetags = join(" ", grep { !m/^subj:/i } split /[\s,]+/, $exclude);
532 my $includesubj = join(" ", map { s/^subj://i; $_ } grep { m/^subj:/i } split /[\s,]+/, $include);
533 my $excludesubj = join(" ", map { s/^subj://i; $_ } grep { m/^subj:/i } split /[\s,]+/, $exclude);
534 my $vismindays = ($mindays == 0 ? "" : $mindays);
535 my $vismaxdays = ($maxdays == -1 ? "" : $maxdays);
536
537 my $sel_rmy = ($repeatmerged ? " selected" : "");
538 my $sel_rmn = ($repeatmerged ? "" : " selected");
539 my $sel_ordraw = ($ordering eq "raw" ? " selected" : "");
540 my $sel_ordold = ($ordering eq "old" ? " selected" : "");
541 my $sel_ordnor = ($ordering eq "normal" ? " selected" : "");
542
543 my $chk_bugrev = ($bug_rev ? " checked" : "");
544 my $chk_pendrev = ($pend_rev ? " checked" : "");
545 my $chk_sevrev = ($sev_rev ? " checked" : "");
546
547 print <<EOF;
548 <tr><td>Only include bugs tagged with </td><td><input name=include value="$includetags"> or that have <input name=includesubj value="$includesubj"> in their subject</td></tr>
549 <tr><td>Exclude bugs tagged with </td><td><input name=exclude value="$excludetags"> or that have <input name=excludesubj value="$excludesubj"> in their subject</td></tr>
550 <tr><td>Only show bugs older than</td><td><input name=mindays value="$vismindays" size=5> days, and younger than <input name=maxdays value="$vismaxdays" size=5> days</td></tr>
551
552 <tr><td>&nbsp;</td></tr>
553
554 </td></tr>
555 <tr><td>Merged bugs should be</td><td>
556 <select name=repeatmerged>
557 <option value=yes$sel_rmy>displayed separately</option>
558 <option value=no$sel_rmn>combined</option>
559 </select>
560 <tr><td>Categorise bugs by</td><td>
561 <select name=ordering>
562 <option value=raw$sel_ordraw>bug number only</option>
563 <option value=old$sel_ordold>status, and severity</option>
564 <option value=normal$sel_ordnor>status, severity and classification</option>
565 EOF
566
567 {
568 my $any = 0;
569 my $o = $param{"ordering"} || "";
570 for my $i (1..$cats) {
571     my $n = get_cat_name($i);
572     next unless defined $n;
573     unless ($any) {
574         $any = 1;
575         print "<option disabled>------</option\n";
576     }
577     printf "<option value=user%s%s>%s</option>\n",
578         $i, ($o eq "user$i" ? " selected" : ""), $n;
579 }
580 }
581
582 print "</select></td></tr>\n";
583
584 printf "<tr><td>Order bugs by</td><td>%s</td></tr>\n",
585     pkg_htmlselectyesno("pend-rev", "outstanding bugs first", "done bugs first", $pend_rev);
586 printf "<tr><td></td><td>%s</td></tr>\n",
587     pkg_htmlselectyesno("sev-rev", "highest severity first", "lowest severity first", $sev_rev);
588 printf "<tr><td></td><td>%s</td></tr>\n",
589     pkg_htmlselectyesno("bug-rev", "oldest bugs first", "newest bugs first", $bug_rev);
590
591 print <<EOF;
592 <tr><td>&nbsp;</td></tr>
593 <tr><td colspan=2><input value="Reload page" type="submit"> with new settings</td></tr>
594 EOF
595
596 print "</table></form></div>\n";
597
598 print "<h2 class=\"outstanding\"><a class=\"options\" href=\"javascript:toggle(2)\">User Categorisations (beta)</a></h2>\n";
599 print "<div id=\"a_2\">\n";
600 print <<EOF;
601 <p>This form allows you to define a new categorisation to use to view bugs
602 against packages. Once defined it will show up as an available category
603 in the Options section above. Note there are a limited numbering of
604 categorisations you can define, so you may need to choose a pre-existing
605 categorisation to replace. Note that this feature currently requires both
606 Javascript and cookies to be enabled. Some usage information is available
607 via Anthony Towns' <a href="http://code.erisian.com.au/Wiki/debbugs/SectioningNotes">development notes</a>.
608 </p>
609 EOF
610
611 printf "<form name=\"categories\" action=\"%s\" method=POST>\n", myurl();
612 print "<table class=\"forms\">\n";
613
614 sub get_cat_name {
615     my $i = shift;
616     if (defined $param{"cat${i}_nam0"}) {
617         my @nams = ();
618         my $j = 0;
619         while (defined $param{"cat${i}_nam$j"}) {
620             push @nams, $param{"cat${i}_nam$j"};
621             $j++;
622         }
623         return join(", ", @nams);
624     } else {
625         return undef;
626     }
627 }
628
629 print "<tr><td>Categorisation to set</td><td>\n";
630 print "<select name=categorisation>\n";
631 my $default = 1;
632 for my $i (1..$cats) {
633     my $name = get_cat_name($i);
634     unless (defined $name) {
635         $name = "(unused)";
636         $default = $i if $default == 0;
637     }
638     printf "<option value=%s%s>%s. %s</option>\n",
639         $i, ($default == $i ? " selected" : ""), $i, $name;
640 }
641 my $defusers = $param{"cat${default}_users"} || $users;
642 print "</select></td></tr>\n";
643 print "<tr><td>Include usertags set by</td><td>\n";
644 print "<input id=users size=50 value=\"$defusers\"></td></tr>\n";
645 print "<tr><td>&nbsp;</td></tr>\n";
646
647 for my $level (0..3) {
648     my $hlevel = $level + 1;
649     my ($n, $s, $t, $o) =
650         map { $param{"cat${default}_${_}${level}"} || "" }
651             ("nam", "pri", "ttl", "ord");
652
653     print <<EOF;
654 <tr><td>Level</td><td>$hlevel</td></tr>
655 <tr><td>Name</td><td><input id="nam$level" value="$n"></td></tr>
656 <tr><td>Sections</td><td><input id="pri$level" value="$s" size=70></td></tr>
657 <tr><td>Titles</td><td><input id="ttl$level" value="$t" size=70></td></tr>
658 <tr><td>Ordering</td><td><input id="ord$level" value="$o" size=20></td></tr>
659 <tr><td>&nbsp;</td></tr>
660 EOF
661 }
662     
663 print <<EOF;
664 <tr><td colspan=2><a href="javascript:save_cat_cookies();">Commit new ordering</a></td></tr>
665 EOF
666
667 print "</table></form></div>\n";
668
669 print "<hr>\n";
670 print "<p>$tail_html";
671
672 print "</body></html>\n";
673
674 sub pkg_htmlindexentrystatus {
675     my $s = shift;
676     my %status = %{$s};
677
678     my $result = "";
679
680     my $showseverity;
681     if  ($status{severity} eq 'normal') {
682         $showseverity = '';
683     } elsif (isstrongseverity($status{severity})) {
684         $showseverity = "Severity: <em class=\"severity\">$status{severity}</em>;\n";
685     } else {
686         $showseverity = "Severity: <em>$status{severity}</em>;\n";
687     }
688
689     $result .= pkg_htmlpackagelinks($status{"package"}, 1);
690
691     my $showversions = '';
692     if (@{$status{found_versions}}) {
693         my @found = @{$status{found_versions}};
694         local $_;
695         s{/}{ } foreach @found;
696         $showversions .= join ', ', map htmlsanit($_), @found;
697     }
698     if (@{$status{fixed_versions}}) {
699         $showversions .= '; ' if length $showversions;
700         $showversions .= '<strong>fixed</strong>: ';
701         my @fixed = @{$status{fixed_versions}};
702         local $_;
703         s{/}{ } foreach @fixed;
704         $showversions .= join ', ', map htmlsanit($_), @fixed;
705     }
706     $result .= " ($showversions)" if length $showversions;
707     $result .= ";\n";
708
709     $result .= $showseverity;
710     $result .= pkg_htmladdresslinks("Reported by: ", \&submitterurl,
711                                 $status{originator});
712     $result .= ";\nOwned by: " . htmlsanit($status{owner})
713                if length $status{owner};
714     $result .= ";\nTags: <strong>" 
715                  . htmlsanit(join(", ", sort(split(/\s+/, $status{tags}))))
716                  . "</strong>"
717                        if (length($status{tags}));
718
719     my @merged= split(/ /,$status{mergedwith});
720     my $mseparator= ";\nMerged with ";
721     for my $m (@merged) {
722         $result .= $mseparator."<A class=\"submitter\" href=\"" . bugurl($m) . "\">#$m</A>";
723         $mseparator= ", ";
724     }
725
726     my $days = 0;
727     if (length($status{done})) {
728         $result .= "<br><strong>Done:</strong> " . htmlsanit($status{done});
729         $days = ceil($debbugs::gRemoveAge - -M buglog($status{id}));
730         if ($days >= 0) {
731             $result .= ";\n<strong>Will be archived" . ( $days == 0 ? " today" : $days == 1 ? " in $days day" : " in $days days" ) . "</strong>";
732         } else {
733             $result .= ";\n<strong>Archived</strong>";
734         }
735     }
736
737     unless (length($status{done})) {
738         if (length($status{forwarded})) {
739             $result .= ";\n<strong>Forwarded</strong> to "
740                        . maybelink($status{forwarded});
741         }
742         my $daysold = int((time - $status{date}) / 86400);   # seconds to days
743         if ($daysold >= 7) {
744             my $font = "";
745             my $efont = "";
746             $font = "em" if ($daysold > 30);
747             $font = "strong" if ($daysold > 60);
748             $efont = "</$font>" if ($font);
749             $font = "<$font>" if ($font);
750
751             my $yearsold = int($daysold / 365);
752             $daysold -= $yearsold * 365;
753
754             $result .= ";\n $font";
755             my @age;
756             push @age, "1 year" if ($yearsold == 1);
757             push @age, "$yearsold years" if ($yearsold > 1);
758             push @age, "1 day" if ($daysold == 1);
759             push @age, "$daysold days" if ($daysold > 1);
760             $result .= join(" and ", @age);
761             $result .= " old$efont";
762         }
763     }
764
765     $result .= ".";
766
767     return $result;
768 }
769
770
771 sub pkg_htmlizebugs {
772     $b = $_[0];
773     my @bugs = @$b;
774
775     my @status = ();
776     my %count;
777     my $header = '';
778     my $footer = "<h2 class=\"outstanding\">Summary</h2>\n";
779
780     my @dummy = ($debbugs::gRemoveAge); #, @debbugs::gSeverityList, @debbugs::gSeverityDisplay);  #, $debbugs::gHTMLExpireNote);
781
782     if (@bugs == 0) {
783         return "<HR><H2>No reports found!</H2></HR>\n";
784     }
785
786     if ( $bug_rev ) {
787         @bugs = sort {$b<=>$a} @bugs;
788     } else {
789         @bugs = sort {$a<=>$b} @bugs;
790     }
791     my %seenmerged;
792
793     my %common = (
794         'show_list_header' => 1,
795         'show_list_footer' => 1,
796     );
797
798     my %section = ();
799
800     foreach my $bug (@bugs) {
801         my %status = %{getbugstatus($bug)};
802         next unless %status;
803         next if bugfilter($bug, %status);
804
805         my $html = sprintf "<li><a href=\"%s\">#%d: %s</a>\n<br>",
806             bugurl($bug), $bug, htmlsanit($status{subject});
807         $html .= pkg_htmlindexentrystatus(\%status) . "\n";
808
809         my $key = "";
810         for my $i (0..$#prior) {
811             my $v = get_bug_order_index($prior[$i], \%status);
812             my $k = $prior[$i]->[$v];
813             $count{"g_${i}_${v}"}++;
814             $key .= "_$v";
815         }
816         $section{$key} .= $html;
817         $count{"_$key"}++;
818
819         push @status, [ $bug, \%status, $html ];
820     }
821
822     my $result = "";
823     if ($ordering eq "raw") {
824         $result .= "<UL class=\"bugs\">\n" . join("", map( { $_->[ 2 ] } @status ) ) . "</UL>\n";
825     } else {
826         $header .= "<ul>\n<div class=\"msgreceived\">\n";
827         my @keys_in_order = ("");
828         for my $o (@order) {
829             push @keys_in_order, "X";
830             while ((my $k = shift @keys_in_order) ne "X") {
831                 for my $k2 (@{$o}) {
832                     push @keys_in_order, "${k}_${k2}";
833                 }
834             }
835         }
836         for ( my $i = 0; $i <= $#keys_in_order; $i++ ) {
837             my $order = $keys_in_order[ $i ];
838             next unless defined $section{$order};
839             my @ttl = split /_/, $order; shift @ttl;
840             my $title = $title[0]->[$ttl[0]] . " bugs";
841             if ($#ttl > 0) {
842                 $title .= " -- ";
843                 $title .= join("; ", grep {$_ ne ""}
844                         map { $title[$_]->[$ttl[$_]] } 1..$#ttl);
845             }
846
847             my $count = $count{"_$order"};
848             my $bugs = $count == 1 ? "bug" : "bugs";
849
850             $header .= "<li><a href=\"#$order\">$title</a> ($count $bugs)</li>\n";
851             if ($common{show_list_header}) {
852                 my $count = $count{"_$order"};
853                 my $bugs = $count == 1 ? "bug" : "bugs";
854                 $result .= "<H2 CLASS=\"outstanding\"><a name=\"$order\"></a>$title ($count $bugs)</H2>\n";
855             } else {
856                 $result .= "<H2 CLASS=\"outstanding\">$title</H2>\n";
857             }
858             $result .= "<div class=\"msgreceived\">\n<UL class=\"bugs\">\n";
859             $result .= "\n\n\n\n";
860             $result .= $section{$order};
861             $result .= "\n\n\n\n";
862             $result .= "</UL>\n</div>\n";
863         } 
864         $header .= "</ul></div>\n";
865
866         $footer .= "<ul>\n<div class=\"msgreceived\">";
867         for my $i (0..$#prior) {
868             my $local_result = '';
869             foreach my $key ( @{$order[$i]} ) {
870                 my $count = $count{"g_${i}_$key"};
871                 next if !$count or !$title[$i]->[$key];
872                 $local_result .= "<li>$count $title[$i]->[$key]</li>\n";
873             }
874             if ( $local_result ) {
875                 $footer .= "<li>$names[$i]<ul>\n$local_result</ul></li>\n";
876             }
877         }
878         $footer .= "</div></ul>\n";
879     }
880
881     $result = $header . $result if ( $common{show_list_header} );
882     $result .= $footer if ( $common{show_list_footer} );
883     return $result;
884 }
885
886 sub pkg_htmlpackagelinks {
887     my $pkgs = shift;
888     return unless defined $pkgs and $pkgs ne '';
889     my $strong = shift;
890     my @pkglist = splitpackages($pkgs);
891
892     $strong = 0;
893     my $openstrong  = $strong ? '<strong>' : '';
894     my $closestrong = $strong ? '</strong>' : '';
895
896     return 'Package' . (@pkglist > 1 ? 's' : '') . ': ' .
897            join(', ',
898                 map {
899                     '<a class="submitter" href="' . pkgurl($_) . '">' .
900                     $openstrong . htmlsanit($_) . $closestrong . '</a>'
901                 } @pkglist
902            );
903 }
904
905 sub pkg_htmladdresslinks {
906     my ($prefixfunc, $urlfunc, $addresses) = @_;
907     if (defined $addresses and $addresses ne '') {
908         my @addrs = getparsedaddrs($addresses);
909         my $prefix = (ref $prefixfunc) ? $prefixfunc->(scalar @addrs)
910                                        : $prefixfunc;
911         return $prefix .
912                join ', ', map { sprintf '<a class="submitter" href="%s">%s</a>',
913                                         $urlfunc->($_->address),
914                                         htmlsanit($_->format) || '(unknown)'
915                               } @addrs;
916     } else {
917         my $prefix = (ref $prefixfunc) ? $prefixfunc->(1) : $prefixfunc;
918         return sprintf '%s<a class="submitter" href="%s">(unknown)</a>', $prefix, $urlfunc->('');
919     }
920 }
921
922 sub pkg_javascript {
923     return <<EOF ;
924 <script type="text/javascript">
925 <!--
926 function pagemain() {
927         toggle(1);
928         toggle(2);
929         enable(1);
930 }
931
932 function setCookie(name, value, expires, path, domain, secure) {
933   var curCookie = name + "=" + escape(value) +
934       ((expires) ? "; expires=" + expires.toGMTString() : "") +
935       ((path) ? "; path=" + path : "") +
936       ((domain) ? "; domain=" + domain : "") +
937       ((secure) ? "; secure" : "");
938   document.cookie = curCookie;
939 }
940
941 function save_cat_cookies() {
942   var cat = document.categories.categorisation.value;
943   var exp = new Date();
944   exp.setTime(exp.getTime() + 10 * 365 * 24 * 60 * 60 * 1000);
945   var oldexp = new Date();
946   oldexp.setTime(oldexp.getTime() - 1 * 365 * 24 * 60 * 60 * 1000);
947   var lev;
948   var done = 0;
949
950   var u = document.getElementById("users");
951   if (u != null) { u = u.value; }
952   if (u == "") { u = null; }
953   if (u != null) {
954       setCookie("cat" + cat + "_users", u, exp, "/");
955   } else {
956       setCookie("cat" + cat + "_users", "", oldexp, "/");
957   }
958
959   var bits = new Array("nam", "pri", "ttl", "ord");
960   for (var i = 0; i < 4; i++) {
961       for (var j = 0; j < bits.length; j++) {
962           var e = document.getElementById(bits[j] + i);
963           if (e) e = e.value;
964           if (e == null) { e = ""; }
965           if (j == 0 && e == "") { done = 1; }
966           if (done || e == "") {
967               setCookie("cat" + cat + "_" + bits[j] + i, "", oldexp, "/");
968           } else {
969               setCookie("cat" + cat + "_" + bits[j] + i, e, exp, "/");
970           }
971       }
972   }
973 }
974
975 function toggle(i) {
976         var a = document.getElementById("a_" + i);
977         if (a.style.display == "none") {
978                 a.style.display = "";
979         } else {
980                 a.style.display = "none";
981         }
982 }
983
984 function enable(x) {
985     for (var i = 1; ; i++) {
986         var a = document.getElementById("b_" + x + "_" + i);
987         if (a == null) break;
988         var ischecked = a.checked;
989         for (var j = 1; ; j++) {
990             var b = document.getElementById("b_" + x + "_"+ i + "_" + j);
991             if (b == null) break;
992             if (ischecked) {
993                 b.disabled = false;
994             } else {
995                 b.disabled = true;
996             }
997         }
998     }
999 }
1000 -->
1001 </script>
1002 EOF
1003 }
1004
1005 sub pkg_htmlselectyesno {
1006     my ($name, $n, $y, $default) = @_;
1007     return sprintf('<select name="%s"><option value=no%s>%s</option><option value=yes%s>%s</option></select>', $name, ($default ? "" : " selected"), $n, ($default ? " selected" : ""), $y);
1008 }
1009
1010 sub pkg_htmlselectsuite {
1011     my $id = sprintf "b_%d_%d_%d", $_[0], $_[1], $_[2];
1012     my @suites = ("stable", "testing", "unstable", "experimental");
1013     my %suiteaka = ("stable", "sarge", "testing", "etch", "unstable", "sid");
1014     my $defaultsuite = "unstable";
1015
1016     my $result = sprintf '<select name=dist id="%s">', $id;
1017     for my $s (@suites) {
1018         $result .= sprintf '<option value="%s"%s>%s%s</option>',
1019                 $s, ($defaultsuite eq $s ? " selected" : ""),
1020                 $s, (defined $suiteaka{$s} ? " (" . $suiteaka{$s} . ")" : "");
1021     }
1022     $result .= '</select>';
1023     return $result;
1024 }
1025
1026 sub pkg_htmlselectarch {
1027     my $id = sprintf "b_%d_%d_%d", $_[0], $_[1], $_[2];
1028     my @arches = qw(alpha amd64 arm hppa i386 ia64 m68k mips mipsel powerpc s390 sparc);
1029
1030     my $result = sprintf '<select name=arch id="%s">', $id;
1031     $result .= '<option value="any">any architecture</option>';
1032     for my $a (@arches) {
1033         $result .= sprintf '<option value="%s">%s</option>', $a, $a;
1034     }
1035     $result .= '</select>';
1036     return $result;
1037 }
1038
1039 sub myurl {
1040     return pkg_etc_url($pkg, "pkg", 0) if defined($pkg);
1041     return pkg_etc_url($src, "src", 0) if defined($src);
1042     return pkg_etc_url($maint, "maint", 0) if defined($maint);
1043     return pkg_etc_url($submitter, "submitter", 0) if defined($submitter);
1044     return pkg_etc_url($severity, "severity", 0) if defined($severity);
1045     return pkg_etc_url($tag, "tag", 0) if defined($tag);
1046 }
1047
1048 sub make_order_list {
1049     my $vfull = shift;
1050     my @x = ();
1051
1052     if ($vfull =~ m/^([^:]+):(.*)$/) {
1053         my $v = $1;
1054         for my $vv (split /,/, $2) {
1055             push @x, "$v=$vv";
1056         }
1057     } else {
1058         for my $v (split /,/, $vfull) {
1059             next unless $v =~ m/.=./;
1060             push @x, $v;
1061         }
1062     }
1063     push @x, "";  # catch all
1064     return @x;
1065 }
1066
1067 sub get_bug_order_index {
1068     my $order = shift;
1069     my $status = shift;
1070     my $pos = -1;
1071
1072     my %tags = ();
1073     %tags = map { $_, 1 } split / /, $status->{"tags"}
1074         if defined $status->{"tags"};
1075
1076     for my $el (@${order}) {
1077         $pos++;
1078         my $match = 1;
1079         for my $item (split /[+]/, $el) {
1080             my ($f, $v) = split /=/, $item, 2;
1081             next unless (defined $f and defined $v);
1082             my $isokay = 0;
1083             $isokay = 1 if (defined $status->{$f} and $v eq $status->{$f});
1084             $isokay = 1 if ($f eq "tag" && defined $tags{$v});
1085             unless ($isokay) {
1086                 $match = 0;
1087                 last;
1088             }
1089         }
1090         if ($match) {
1091             return $pos;
1092             last;
1093         }
1094     }
1095     return -1;
1096 }
1097