[BACK]Return to cvsweb.cgi CVS log [TXT][DIR] Up to [cvsweb.bsd.lv] / cvsweb

Diff for /cvsweb/cvsweb.cgi between version 1.1.1.16 and 3.35

version 1.1.1.16, 2000/12/28 18:37:25 version 3.35, 2000/10/10 21:14:05
Line 42 
Line 42 
 # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF  # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 # SUCH DAMAGE.  # SUCH DAMAGE.
 #  #
 # $zId: cvsweb.cgi,v 1.104 2000/11/01 22:05:12 hnordstrom Exp $  # $zId: cvsweb.cgi,v 1.103 2000/09/20 17:02:29 jumager Exp $
 # $kId: cvsweb.cgi,v 1.47 2000/12/28 18:07:20 knu Exp $  # $Id$
 #  #
 ###  ###
   
Line 52  use strict;
Line 52  use strict;
 use vars qw (  use vars qw (
     $config $allow_version_select $verbose      $config $allow_version_select $verbose
     %CVSROOT %CVSROOTdescr %MIRRORS %DEFAULTVALUE %ICONS %MTYPES      %CVSROOT %CVSROOTdescr %MIRRORS %DEFAULTVALUE %ICONS %MTYPES
     @DIFFTYPES %DIFFTYPES @LOGSORTKEYS %LOGSORTKEYS  
     %alltags @tabcolors %fileinfo %tags @branchnames %nameprinted      %alltags @tabcolors %fileinfo %tags @branchnames %nameprinted
     %symrev %revsym @allrevisions %date %author @revdisplayorder      %symrev %revsym @allrevisions %date %author @revdisplayorder
     @revisions %state %difflines %log %branchpoint @revorder      @revisions %state %difflines %log %branchpoint @revorder
Line 62  use vars qw (
Line 61  use vars qw (
     %funcline_regexp $is_mod_perl      %funcline_regexp $is_mod_perl
     $is_links $is_lynx $is_w3m $is_msie $is_mozilla3 $is_textbased      $is_links $is_lynx $is_w3m $is_msie $is_mozilla3 $is_textbased
     %input $query $barequery $sortby $bydate $byrev $byauthor      %input $query $barequery $sortby $bydate $byrev $byauthor
     $bylog $byfile $defaultDiffType $logsort $cvstree $cvsroot      $bylog $byfile $hr_default $logsort $cvstree $cvsroot
     $mimetype $charset $defaultTextPlain $defaultViewable      $mimetype $defaultTextPlain $defaultViewable $allow_compress
     $allow_compress $GZIPBIN $backicon $diricon $fileicon      $GZIPBIN $backicon $diricon $fileicon $fullname $newname
     $fullname $newname $cvstreedefault      $cvstreedefault $body_tag $body_tag_for_src
     $body_tag $body_tag_for_src $logo $defaulttitle $address      $logo $defaulttitle $address
     $long_intro $short_instruction $shortLogLen      $long_intro $short_instruction $shortLogLen
     $show_author $dirtable $tablepadding $columnHeaderColorDefault      $show_author $dirtable $tablepadding $columnHeaderColorDefault
     $columnHeaderColorSorted $hr_breakable $showfunc $hr_ignwhite      $columnHeaderColorSorted $hr_breakable $showfunc $hr_ignwhite
Line 79  use vars qw (
Line 78  use vars qw (
     $navigationHeaderColor $tableBorderColor $markupLogColor      $navigationHeaderColor $tableBorderColor $markupLogColor
     $tabstop $state $annTable $sel $curbranch @HideModules      $tabstop $state $annTable $sel $curbranch @HideModules
     $module $use_descriptions %descriptions @mytz $dwhere $moddate      $module $use_descriptions %descriptions @mytz $dwhere $moddate
     $use_moddate $has_zlib $gzip_open $allow_tar      $use_moddate $has_zlib $gzip_open
     $LOG_FILESEPARATOR $LOG_REVSEPARATOR      $LOG_FILESEPARATOR $LOG_REVSEPARATOR
 );  );
   
 sub printDiffSelect($);  sub printDiffSelect($);
 sub printDiffLinks($$);  
 sub printLogSortSelect($);  
 sub findLastModifiedSubdirs(@);  sub findLastModifiedSubdirs(@);
 sub htmlify_sub(&$);  
 sub htmlify($;$);  sub htmlify($;$);
 sub spacedHtmlText($;$);  sub spacedHtmlText($;$);
 sub link($$);  sub link($$);
Line 121  sub download_link($$$;$);
Line 117  sub download_link($$$;$);
 sub toggleQuery($$);  sub toggleQuery($$);
 sub urlencode($);  sub urlencode($);
 sub htmlquote($);  sub htmlquote($);
 sub htmlunquote($);  
 sub http_header(;$);  sub http_header(;$);
 sub html_header($);  sub html_header($);
 sub html_footer();  sub html_footer();
Line 169  $tabstop = $use_moddate = $moddate = $gzip_open = unde
Line 164  $tabstop = $use_moddate = $moddate = $gzip_open = unde
 $LOG_FILESEPARATOR = q/^={77}$/;  $LOG_FILESEPARATOR = q/^={77}$/;
 $LOG_REVSEPARATOR = q/^-{28}$/;  $LOG_REVSEPARATOR = q/^-{28}$/;
   
 @DIFFTYPES = qw(h H u c s);  
 @DIFFTYPES{@DIFFTYPES} = (  
                           {  
                            'descr'   => 'colored',  
                            'opts'    => [ '-u' ],  
                            'colored' => 1,  
                           },  
                           {  
                            'descr'   => 'long colored',  
                            'opts'    => [ '--unified=15' ],  
                            'colored' => 1,  
                           },  
                           {  
                            'descr'   => 'unified',  
                            'opts'    => [ '-u' ],  
                            'colored' => 0,  
                           },  
                           {  
                            'descr'   => 'context',  
                            'opts'    => [ '-c' ],  
                            'colored' => 0,  
                           },  
                           {  
                            'descr'   => 'side by side',  
                            'opts'    => [ '--side-by-side', '--width=164' ],  
                            'colored' => 0,  
                           },  
                          );  
   
 @LOGSORTKEYS = qw(cvs date rev);  
 @LOGSORTKEYS{@LOGSORTKEYS} = (  
                               {  
                                'descr' => 'Not sorted',  
                               },  
                               {  
                                'descr' => 'Commit date',  
                               },  
                               {  
                                'descr' => 'Revision',  
                               },  
                              );  
   
   
 ##### End of configuration variables #####  ##### End of configuration variables #####
   
 use Time::Local;  use Time::Local;
Line 228  $verbose = $v;
Line 180  $verbose = $v;
 $checkoutMagic = "~checkout~";  $checkoutMagic = "~checkout~";
 $pathinfo = defined($ENV{PATH_INFO}) ? $ENV{PATH_INFO} : '';  $pathinfo = defined($ENV{PATH_INFO}) ? $ENV{PATH_INFO} : '';
 $where = $pathinfo;  $where = $pathinfo;
 $where =~ tr|/|/|s;  
 $doCheckout = ($where =~ /^\/$checkoutMagic/);  $doCheckout = ($where =~ /^\/$checkoutMagic/);
 $where =~ s|^/($checkoutMagic)?||;  $where =~ s|^/($checkoutMagic)?||;
 $where =~ s|/$||;  $where =~ s|/+$||;
 $scriptname = defined($ENV{SCRIPT_NAME}) ? $ENV{SCRIPT_NAME} : '';  $scriptname = defined($ENV{SCRIPT_NAME}) ? $ENV{SCRIPT_NAME} : '';
 $scriptname =~ s|^/?|/|;  $scriptname =~ s|^/?|/|;
 $scriptname =~ s|/+$||;  $scriptname =~ s|/+$||;
Line 245  $is_mod_perl = defined($ENV{MOD_PERL});
Line 196  $is_mod_perl = defined($ENV{MOD_PERL});
 # in lynx, it it very annoying to have two links  # in lynx, it it very annoying to have two links
 # per file, so disable the link at the icon  # per file, so disable the link at the icon
 # in this case:  # in this case:
 $Browser = $ENV{HTTP_USER_AGENT} || '';  $Browser = $ENV{HTTP_USER_AGENT};
 $is_links = ($Browser =~ m`^Links `);  $is_links = ($Browser =~ m`^Links `);
 $is_lynx = ($Browser =~ m`^Lynx/`i);  $is_lynx = ($Browser =~ m`^Lynx/`i);
 $is_w3m = ($Browser =~ m`^w3m/`i);  $is_w3m = ($Browser =~ m`^w3m/`i);
Line 295  $query = $ENV{QUERY_STRING};
Line 246  $query = $ENV{QUERY_STRING};
   
 if (defined($query) && $query ne '') {  if (defined($query) && $query ne '') {
     foreach (split(/&/, $query)) {      foreach (split(/&/, $query)) {
         y/+/ /;  
         s/%(..)/sprintf("%c", hex($1))/ge;      # unquote %-quoted          s/%(..)/sprintf("%c", hex($1))/ge;      # unquote %-quoted
         if (/(\S+)=(.*)/) {          if (/(\S+)=(.*)/) {
             $input{$1} = $2 if ($2 ne "");              $input{$1} = $2 if ($2 ne "");
Line 377  else {
Line 327  else {
     $byfile = 1;      $byfile = 1;
 }  }
   
 $defaultDiffType = $input{'f'};  $hr_default = $input{'f'} eq 'h';
   
 $logsort = $input{'logsort'};  $logsort = $input{'logsort'};
   
Line 406  foreach $k (keys %ICONS) {
Line 356  foreach $k (keys %ICONS) {
     my ($itxt,$ipath,$iwidth,$iheight) = @{$ICONS{$k}};      my ($itxt,$ipath,$iwidth,$iheight) = @{$ICONS{$k}};
     if ($ipath) {      if ($ipath) {
         ${"${k}icon"} = sprintf('<IMG SRC="%s" ALT="%s" BORDER="0" WIDTH="%d" HEIGHT="%d">',          ${"${k}icon"} = sprintf('<IMG SRC="%s" ALT="%s" BORDER="0" WIDTH="%d" HEIGHT="%d">',
                                 hrefquote($ipath), htmlquote($itxt), $iwidth, $iheight)                                  htmlquote($ipath), htmlquote($itxt), $iwidth, $iheight)
     }      }
     else {      else {
         ${"${k}icon"} = $itxt;          ${"${k}icon"} = $itxt;
Line 473  $module = $1;
Line 423  $module = $1;
 if ($module && &forbidden_module($module)) {  if ($module && &forbidden_module($module)) {
     &fatal("403 Forbidden", "Access to $where forbidden.");      &fatal("403 Forbidden", "Access to $where forbidden.");
 }  }
   
 #  
 # Handle tarball downloads before any headers are output.  
 #  
 if ($input{tarball}) {  
     &fatal("403 Forbidden", "Downloading tarballs is prohibited.")  
       unless $allow_tar;  
     $where =~ s,/[^/]*$,,;  
     $where =~ s,^/,,;  
     my($basedir) = ($where =~ m,([^/]+)$,);  
   
     if ($basedir eq '' || $where eq '') {  
         &fatal("500 Internal Error", "You cannot download the top level directory.");  
     }  
   
     my $tmpdir = "/tmp/.cvsweb.$$." . int(time);  
   
     mkdir($tmpdir, 0700)  
       or &fatal("500 Internal Error", "Unable to make temporary directory: $!");  
   
     my $fatal = '';  
   
     do {  
         chdir $tmpdir  
           or $fatal = "500 Internal Error", "Unable to cd to temporary directory: $!"  
             && last;  
   
         my @params = (exists $input{only_with_tag} && length $input{only_with_tag})  
           ? ("-r", $input{only_with_tag}) : ();  
   
         system "cvs", "-RlQd", $cvsroot, "co", @params, $where  
           and $fatal = "500 Internal Error","cvs co failure: $!: $where"  
             && last;  
   
         chdir "$where/.."  
           or $fatal = "500 Internal Error","Cannot find expected directory in checkout"  
             && last;  
   
         $| = 1; # Essential to get the buffering right.  
   
         print "Content-type: application/x-gzip\r\n\r\n";  
   
         system "tar", "--ignore-failed-read", "--exclude", "CVS", "-zcf", "-", $basedir  
           and $fatal = "500 Internal Error","tar zc failure: $!: $basedir"  
             && last;  
   
         chdir $tmpdir  
           or $fatal = "500 Internal Error","Unable to cd to temporary directory: $!"  
             && last;  
     } while (0);  
   
     system "rm", "-rf", $tmpdir if -d $tmpdir;  
   
     &fatal($fatal) if $fatal;  
   
     exit;  
 }  
   
 ##############################  ##############################
 # View a directory  # View a directory
 ###############################  ###############################
 if (-d $fullname) {  elsif (-d $fullname) {
         my $dh = do {local(*DH);};          my $dh = do {local(*DH);};
         opendir($dh, $fullname) || &fatal("404 Not Found","$where: $!");          opendir($dh, $fullname) || &fatal("404 Not Found","$where: $!");
         my @dir = readdir($dh);          my @dir = readdir($dh);
Line 590  if (-d $fullname) {
Line 482  if (-d $fullname) {
             }              }
             print "<table  width=\"100%\" border=0 cellspacing=1 cellpadding=$tablepadding>\n";              print "<table  width=\"100%\" border=0 cellspacing=1 cellpadding=$tablepadding>\n";
             $infocols++;              $infocols++;
             printf '<tr><th align=left bgcolor="%s">',              print "<tr><th align=left bgcolor=\"" . (($byfile) ?
               $byfile ? $columnHeaderColorSorted : $columnHeaderColorDefault;                                                     $columnHeaderColorSorted :
                                                      $columnHeaderColorDefault) . "\">";
             if ($byfile) {              if ($byfile) {
                 print 'File';                  print 'File';
             } else {              } else {
Line 603  if (-d $fullname) {
Line 496  if (-d $fullname) {
             # with revision information:              # with revision information:
             if (scalar(%fileinfo)) {              if (scalar(%fileinfo)) {
                 $infocols++;                  $infocols++;
                 printf '<th align=left bgcolor="%s">',                  print "<th align=left bgcolor=\"" . (($byrev) ?
                   $byrev ? $columnHeaderColorSorted : $columnHeaderColorDefault;                                                     $columnHeaderColorSorted :
                                                      $columnHeaderColorDefault) . "\">";
                 if ($byrev) {                  if ($byrev) {
                     print 'Rev.';                      print 'Rev.';
                 } else {                  } else {
Line 613  if (-d $fullname) {
Line 507  if (-d $fullname) {
                 }                  }
                 print "</th>";                  print "</th>";
                 $infocols++;                  $infocols++;
                 printf '<th align=left bgcolor="%s">',                  print "<th align=left bgcolor=\"" . (($bydate) ?
                   $bydate ? $columnHeaderColorSorted : $columnHeaderColorDefault;                                                     $columnHeaderColorSorted :
                                                      $columnHeaderColorDefault) . "\">";
                 if ($bydate) {                  if ($bydate) {
                     print 'Age';                      print 'Age';
                 } else {                  } else {
                     print &link('Age', sprintf('./%s#dirlist',                      print &link('Age', sprintf('./%s#dirlist',
                                                &toggleQuery("sortby", "date")));                                                  &toggleQuery("sortby", "date")));
                 }                  }
                 print "</th>";                  print "</th>";
                 if ($show_author) {                  if ($show_author) {
                     $infocols++;                      $infocols++;
                     printf '<th align=left bgcolor="%s">',                      print "<th align=left bgcolor=\"" . (($byauthor) ?
                       $byauthor ? $columnHeaderColorSorted : $columnHeaderColorDefault;                                                     $columnHeaderColorSorted :
                                                      $columnHeaderColorDefault) . "\">";
                     if ($byauthor) {                      if ($byauthor) {
                         print 'Author';                          print 'Author';
                     } else {                      } else {
Line 635  if (-d $fullname) {
Line 531  if (-d $fullname) {
                     print "</th>";                      print "</th>";
                 }                  }
                 $infocols++;                  $infocols++;
                 printf '<th align=left bgcolor="%s">',                  print "<th align=left bgcolor=\"" . (($bylog) ?
                   $bylog ? $columnHeaderColorSorted : $columnHeaderColorDefault;                                                 $columnHeaderColorSorted :
                                                  $columnHeaderColorDefault) . "\">";
                 if ($bylog) {                  if ($bylog) {
                     print 'Last log entry';                      print 'Last log entry';
                 } else {                  } else {
                     print &link('Last log entry', sprintf('./%s#dirlist',                      print &link('Last log entry', sprintf('./%s#dirlist',
                                                           &toggleQuery("sortby", "log")));                                                    &toggleQuery("sortby", "log")));
                 }                  }
                 print "</th>";                  print "</th>";
             }              }
             elsif ($use_descriptions) {              elsif ($use_descriptions) {
                 printf '<th align=left bgcolor="%s">', $columnHeaderColorDefault;                  print "<th align=left bgcolor=\"". $columnHeaderColorDefault . "\">";
                 print "Description";                  print "Description";
                 $infocols++;                  $infocols++;
             }              }
Line 711  if (-d $fullname) {
Line 608  if (-d $fullname) {
                 next if ($_ eq '..' && $where eq '/');                  next if ($_ eq '..' && $where eq '/');
                 my ($rev,$date,$log,$author,$filename) = @{$fileinfo{$_}}                  my ($rev,$date,$log,$author,$filename) = @{$fileinfo{$_}}
                     if (defined($fileinfo{$_}));                      if (defined($fileinfo{$_}));
                 printf '<tr bgcolor="%s"><td>', $tabcolors[$dirrow % 2] if $dirtable;                  print "<tr bgcolor=\"" . @tabcolors[$dirrow%2] . "\"><td>" if ($dirtable);
                 if ($_ eq '..') {                  if ($_ eq '..') {
                     $url = "../$query";                      $url = "../" . $query;
                     if ($nofilelinks) {                      if ($nofilelinks) {
                         print $backicon;                          print $backicon;
                     }                      }
Line 723  if (-d $fullname) {
Line 620  if (-d $fullname) {
                     print " ", &link("Previous Directory", $url);                      print " ", &link("Previous Directory", $url);
                 }                  }
                 else {                  else {
                     $url = urlencode($_) . "/$query";                      $url = urlencode($_) . '/' . $query;
                     print "<A NAME=\"$_\"></A>";                      print "<A NAME=\"$_\"></A>";
                     if ($nofilelinks) {                      if ($nofilelinks) {
                         print $diricon;                          print $diricon;
Line 742  if (-d $fullname) {
Line 639  if (-d $fullname) {
                 if ($filename) {                  if ($filename) {
                     print "</td><td>&nbsp;</td><td>&nbsp;" if ($dirtable);                      print "</td><td>&nbsp;</td><td>&nbsp;" if ($dirtable);
                     if ($date) {                      if ($date) {
                         print " <i>", readableTime(time() - $date,0), "</i>";                          print " <i>" . readableTime(time() - $date,0) . "</i>";
                     }                      }
                     if ($show_author) {                      if ($show_author) {
                         print "</td><td>&nbsp;" if ($dirtable);                          print "</td><td>&nbsp;" if ($dirtable);
Line 753  if (-d $fullname) {
Line 650  if (-d $fullname) {
                     print "$filename/$rev";                      print "$filename/$rev";
                     print "<BR>" if ($dirtable);                      print "<BR>" if ($dirtable);
                     if ($log) {                      if ($log) {
                         print "&nbsp;<font size=-1>",                          print "&nbsp;<font size=-1>"
                           &htmlify(substr($log,0,$shortLogLen));                              . &htmlify(substr($log,0,$shortLogLen));
                         if (length $log > 80) {                          if (length $log > 80) {
                             print "...";                              print "...";
                         }                          }
Line 764  if (-d $fullname) {
Line 661  if (-d $fullname) {
                 else {                  else {
                     my ($dwhere) = ($where ne "/" ? $where : "") . $_;                      my ($dwhere) = ($where ne "/" ? $where : "") . $_;
                     if ($use_descriptions && defined $descriptions{$dwhere}) {                      if ($use_descriptions && defined $descriptions{$dwhere}) {
                         print "<TD COLSPAN=", ($infocols-1), ">&nbsp;" if $dirtable;                          print "<TD COLSPAN=" . ($infocols-1) . ">&nbsp;" if $dirtable;
                         print $descriptions{$dwhere};                          print $descriptions{$dwhere};
                     } elsif ($dirtable && $infocols > 1) {                      } elsif ($dirtable && $infocols > 1) {
                         # close the row with the appropriate number of                          # close the row with the appropriate number of
Line 795  if (-d $fullname) {
Line 692  if (-d $fullname) {
                 next if (!defined($fileinfo{$_}));                  next if (!defined($fileinfo{$_}));
                 ($rev,$date,$log,$author) = @{$fileinfo{$_}};                  ($rev,$date,$log,$author) = @{$fileinfo{$_}};
                 $filesfound++;                  $filesfound++;
                 printf '<tr bgcolor="%s"><td>', $tabcolors[$dirrow % 2] if $dirtable;                  print "<tr bgcolor=\"" . @tabcolors[$dirrow%2] . "\"><td>" if ($dirtable);
                 print "<A NAME=\"$_\"></A>";                  print "<A NAME=\"$_\"></A>";
                 if ($nofilelinks) {                  if ($nofilelinks) {
                     print $fileicon;                      print $fileicon;
Line 810  if (-d $fullname) {
Line 707  if (-d $fullname) {
                               $defaultViewable ? "text/x-cvsweb-markup" : undef);                                $defaultViewable ? "text/x-cvsweb-markup" : undef);
                 print "</td><td>&nbsp;" if ($dirtable);                  print "</td><td>&nbsp;" if ($dirtable);
                 if ($date) {                  if ($date) {
                     print " <i>", readableTime(time() - $date,0), "</i>";                      print " <i>" . readableTime(time() - $date,0) . "</i>";
                 }                  }
                 if ($show_author) {                  if ($show_author) {
                     print "</td><td>&nbsp;" if ($dirtable);                      print "</td><td>&nbsp;" if ($dirtable);
Line 818  if (-d $fullname) {
Line 715  if (-d $fullname) {
                 }                  }
                 print "</td><td>&nbsp;" if ($dirtable);                  print "</td><td>&nbsp;" if ($dirtable);
                 if ($log) {                  if ($log) {
                     print " <font size=-1>", &htmlify(substr($log,0,$shortLogLen));                      print " <font size=-1>" . &htmlify(substr($log,0,$shortLogLen));
                     if (length $log > 80) {                      if (length $log > 80) {
                         print "...";                          print "...";
                     }                      }
Line 833  if (-d $fullname) {
Line 730  if (-d $fullname) {
         if ($dirtable && defined($tableBorderColor)) {          if ($dirtable && defined($tableBorderColor)) {
             print "</td></tr></table>";              print "</td></tr></table>";
         }          }
         print( $dirtable == 1 ? "</table>\n" : "</menu>\n" );          print "". ($dirtable == 1) ? "</table>" : "</menu>" . "\n";
   
         if ($filesexists && !$filesfound) {          if ($filesexists && !$filesfound) {
             print "<P><B>NOTE:</B> There are $filesexists files, but none matches the current tag ($input{only_with_tag})\n";              print "<P><B>NOTE:</B> There are $filesexists files, but none matches the current tag ($input{only_with_tag})\n";
Line 872  if (-d $fullname) {
Line 769  if (-d $fullname) {
             print "<INPUT TYPE=SUBMIT VALUE=\"Go\">\n";              print "<INPUT TYPE=SUBMIT VALUE=\"Go\">\n";
             print "</FORM>\n";              print "</FORM>\n";
         }          }
   
         if ($allow_tar) {  
             my($basefile) = ($where =~ m,(?:.*/)?([^/]+),);  
   
             if ($basefile ne '') {  
                 print "<HR NOSHADE>\n",  
                   "<DIV align=center>",  
                     &link("Download this directory in tarball",  
                           # Mangle the filename so browsers show a reasonable  
                           # filename to download.  
                           "$basefile.tar.gz$query".  
                           ($query ? "&" : "?")."tarball=1"),  
                             "</DIV>";  
             }  
         }  
   
         my $formwhere = $scriptwhere;          my $formwhere = $scriptwhere;
         $formwhere =~ s|Attic/?$|| if ($input{'hideattic'});          $formwhere =~ s|Attic/?$|| if ($input{'hideattic'});
   
Line 907  if (-d $fullname) {
Line 788  if (-d $fullname) {
             print "<OPTION",$byrev ? " SELECTED" : ""," VALUE=rev>Revision";              print "<OPTION",$byrev ? " SELECTED" : ""," VALUE=rev>Revision";
             print "<OPTION",$bylog ? " SELECTED" : ""," VALUE=log>Log message";              print "<OPTION",$bylog ? " SELECTED" : ""," VALUE=log>Log message";
             print "</SELECT></td>";              print "</SELECT></td>";
             print "<td>Sort log by: ";              print "<td>revisions by: \n";
             printLogSortSelect(0);              print "<SELECT NAME=logsort>\n";
             print "</td></tr>";              print "<OPTION VALUE=cvs",$logsort eq "cvs" ? " SELECTED" : "", ">Not sorted";
               print "<OPTION VALUE=date",$logsort eq "date" ? " SELECTED" : "", ">Commit date";
               print "<OPTION VALUE=rev",$logsort eq "rev" ? " SELECTED" : "", ">Revision";
               print "</SELECT></td></tr>";
             print "<tr><td>Diff format: ";              print "<tr><td>Diff format: ";
             printDiffSelect(0);              printDiffSelect(0);
             print "</td>";              print "</td>";
Line 985  if (-d $fullname) {
Line 869  if (-d $fullname) {
         my $fh = do {local(*FH);};          my $fh = do {local(*FH);};
         my ($xtra, $module);          my ($xtra, $module);
         # Assume it's a module name with a potential path following it.          # Assume it's a module name with a potential path following it.
         $xtra = (($module = $where) =~ s|/.*||) ? $& : '';          $xtra = $& if (($module = $where) =~ s|/.*||);
         # Is there an indexed version of modules?          # Is there an indexed version of modules?
         if (open($fh, "$cvsroot/CVSROOT/modules")) {          if (open($fh, "$cvsroot/CVSROOT/modules")) {
             while (<$fh>) {              while (<$fh>) {
                 if (/^(\S+)\s+(\S+)/o && $module eq $1                  if (/^(\S+)\s+(\S+)/o && $module eq $1
                     && -d "$cvsroot/$2" && $module ne $2) {                      && -d "${cvsroot}/$2" && $module ne $2) {
                     &redirect("$scriptname/$2$xtra");                      &redirect($scriptname . '/' . $2 . $xtra);
                 }                  }
             }              }
         }          }
Line 1003  gzipclose();
Line 887  gzipclose();
   
 sub printDiffSelect($) {  sub printDiffSelect($) {
     my ($use_java_script) = @_;      my ($use_java_script) = @_;
     my $f = $input{'f'};      my ($f) = $input{'f'};
       print "<SELECT NAME=\"f\"";
     print '<SELECT NAME="f"';      print " onchange=\"submit()\"" if ($use_java_script);
     print ' onchange="submit()"' if $use_java_script;      print ">\n";
     print '>';      print "<OPTION VALUE=h",$f eq "h" ? " SELECTED" : "", ">Colored Diff";
       print "<OPTION VALUE=H",$f eq "H" ? " SELECTED" : "", ">Long Colored Diff";
     local $_;      print "<OPTION VALUE=u",$f eq "u" ? " SELECTED" : "", ">Unidiff";
     for (@DIFFTYPES) {      print "<OPTION VALUE=c",$f eq "c" ? " SELECTED" : "", ">Context Diff";
         printf('<OPTION VALUE="%s"%s>%s',      print "<OPTION VALUE=s",$f eq "s" ? " SELECTED" : "", ">Side by Side";
                $_,  
                $f eq $_ ? ' SELECTED' : '',  
                "\u$DIFFTYPES{$_}{'descr'}"  
               );  
     }  
   
     print "</SELECT>";      print "</SELECT>";
 }  }
   
 sub printLogSortSelect($) {  
     my ($use_java_script) = @_;  
   
     print '<SELECT NAME="logsort"';  
     print ' onchange="submit()"' if $use_java_script;  
     print '>';  
   
     local $_;  
     for (@LOGSORTKEYS) {  
         printf('<OPTION VALUE="%s"%s>%s',  
                $_,  
                $logsort eq $_ ? ' SELECTED' : '',  
                "\u$LOGSORTKEYS{$_}{'descr'}"  
               );  
     }  
   
     print "</SELECT>";  
 }  
   
 sub findLastModifiedSubdirs(@) {  sub findLastModifiedSubdirs(@) {
     my (@dirs) = @_;      my (@dirs) = @_;
     my ($dirname, @files);      my ($dirname, @files);
Line 1074  sub findLastModifiedSubdirs(@) {
Line 933  sub findLastModifiedSubdirs(@) {
     return @files;      return @files;
 }  }
   
 sub htmlify_sub(&$) {  
     (my $proc, local $_) = @_;  
     local @_ = split(m`(<a [^>]+>[^<]*</a>)`i);  
     my $linked;  
     my $result = '';  
   
     while (($_, $linked) = splice(@_, 0, 2)) {  
         &$proc();  
         $result .= $_ if defined($_);  
         $result .= $linked if defined($linked);  
     }  
   
     $result;  
 }  
   
 sub htmlify($;$) {  sub htmlify($;$) {
     (local $_, my $extra) = @_;          (local $_, my $extra) = @_;
   
     $_ = htmlquote($_);          $_ = htmlquote($_);
   
     # get URL's as link          # get URL's as link
     s{          s`(http|ftp|https)(://[-a-zA-Z0-9%.~:_/]+)([?&]([-a-zA-Z0-9%.~:_]+)=([-a-zA-Z0-9%.~:_])+)*`&link("$1$2$3", "$1$2$3")`ge;        # `
       (http|ftp|https)://\S+          # get e-mails as link
      }{          s`[-a-zA-Z0-9_.]+@([-a-zA-Z0-9]+\.)+[A-Za-z]{2,4}`&link($&, "mailto:$&")`ge;    # `
          &link($&, htmlunquote($&))  
      }egx;  
   
     # get e-mails as link          if ($extra) {
     $_ = htmlify_sub {              # get PR #'s as link: "PR#nnnn" "PR: nnnn, ..." "PR nnnn, ..." "bin/nnnn"
         s<              if (defined($prcgi)) {
           [\w+=\-.!]+@[\w\-]+(\.[\w\-]+)+                  1 while s`\b(pr[:#]?\s*(?:#?\d+[,\s]\s*)*#?)(\d+)\b`$1 . &link($2, sprintf($prcgi, $2))`ie; # `;
             ><                  s`\b${prcategories}/(\d+)\b`&link($&, sprintf($prcgi, $1))`igeo;        # `;
               &link($&, "mailto:$&")              }
                 >egix;  
     } $_;  
   
     if ($extra) {              # get manpage specs as link: "foo.1" "foo(1)"
         # get PR #'s as link: "PR#nnnn" "PR: nnnn, ..." "PR nnnn, ..." "bin/nnnn"              if (defined($mancgi)) {
         if (defined($prcgi)) {                  s`\b([a-zA-Z]\w+)(?:\(([0-9n])\)\B|\.([0-9n])\b)`&link($&, sprintf($mancgi, $2 ne '' ? $2 : $3, $1))`ge; # `x;
             my $prev;              }
   
             do {  
                 $prev = $_;  
   
                 $_ = htmlify_sub {  
                     s{  
                       (\bPR[:\#]?\s*  
                        (?:  
                         \#?  
                         \d+[,\s]\s*  
                        )*  
                        \#?)  
                       (\d+)\b  
                      }{  
                          $1 . &link($2, sprintf($prcgi, $2))  
                      }egix;  
                 } $_;  
             } while ($_ ne $prev);  
   
             $_ = htmlify_sub {  
                 s{  
                   (\b$prcategories/(\d+)\b)  
                  }{  
                      &link($1, sprintf($prcgi, $2))  
                  }egox;  
             } $_;  
         }          }
   
         # get manpage specs as link: "foo.1" "foo(1)"          $_;
         if (defined($mancgi)) {  
             $_ = htmlify_sub {  
                 s{  
                   (\b([a-zA-Z][\w_.]+)  
                    (?:  
                     \( ([0-9n]) \)\B  
                     |  
                     \.([0-9n])\b  
                    )  
                   )  
                  }{  
                      &link($1, sprintf($mancgi, $3 ne '' ? $3 : $4, $2))  
                  }egx;  
             } $_;  
         }  
     }  
   
     $_;  
 }  }
   
 sub spacedHtmlText($;$) {  sub spacedHtmlText($;$) {
Line 1196  sub spacedHtmlText($;$) {
Line 992  sub spacedHtmlText($;$) {
 sub link($$) {  sub link($$) {
         my($name, $where) = @_;          my($name, $where) = @_;
   
         sprintf '<A HREF="%s">%s</A>', hrefquote($where), $name;          sprintf '<A HREF="%s">%s</A>', htmlquote($where), $name;
 }  }
   
 sub revcmp($$) {  sub revcmp($$) {
Line 1270  sub safeglob($) {
Line 1066  sub safeglob($) {
                 $glob =~ s/{([^}]+)}/($t = $1) =~ s-,-|-g; "($t)"/eg;                  $glob =~ s/{([^}]+)}/($t = $1) =~ s-,-|-g; "($t)"/eg;
                 foreach (readdir($dh)) {                  foreach (readdir($dh)) {
                         if (/^${glob}$/) {                          if (/^${glob}$/) {
                                 push(@results, "$dirname/" .$_);                                  push(@results, $dirname . "/" .$_);
                         }                          }
                 }                  }
         }          }
Line 1469  sub doAnnotate($$) {
Line 1265  sub doAnnotate($$) {
             }              }
             else {              else {
                 $revprint = sprintf('%-8s', $lrev);                  $revprint = sprintf('%-8s', $lrev);
                 $revprint =~ s`\S+`&link($&, "$scriptwhere$query#rev$&")`e;     # `                  $revprint =~ s`\S+`&link($&, "$scriptwhere$barequery#rev$&")`e; # `
                 $oldLusr = '';                  $oldLusr = '';
             }              }
             if ($lusr eq $oldLusr) {              if ($lusr eq $oldLusr) {
Line 1500  sub doAnnotate($$) {
Line 1296  sub doAnnotate($$) {
             # CVS command line client.  But for simplicity, we don't.              # CVS command line client.  But for simplicity, we don't.
         }          }
         elsif ($words[0] eq "error") {          elsif ($words[0] eq "error") {
             fatal("500 Internal Error", "Error occured during annotate: <b>$_</b>");              fatal ("500 Internal Error", "Error occured during annotate: <b>$_</b>");
         }          }
     }      }
     if ($annTable) {      if ($annTable) {
Line 1565  sub doCheckout($$) {
Line 1361  sub doCheckout($$) {
       open(STDERR, ">&STDOUT"); # Redirect stderr to stdout        open(STDERR, ">&STDOUT"); # Redirect stderr to stdout
       exec("cvs", "-Rld", $cvsroot, "co", "-p", $revopt, $where);        exec("cvs", "-Rld", $cvsroot, "co", "-p", $revopt, $where);
     }      }
   
     if (eof($fh)) {  
         &fatal("404 Not Found",  
                "$where is not (any longer) pertinent");  
     }  
 #===================================================================  #===================================================================
 #Checking out squid/src/ftp.c  #Checking out squid/src/ftp.c
 #RCS:  /usr/src/CVS/squid/src/ftp.c,v  #RCS:  /usr/src/CVS/squid/src/ftp.c,v
Line 1589  sub doCheckout($$) {
Line 1380  sub doCheckout($$) {
     }      }
     if ($filename ne $where) {      if ($filename ne $where) {
         &fatal("500 Internal Error",          &fatal("500 Internal Error",
                "Unexpected output from cvs co: $cvsheader");                 "Unexpected output from cvs co: $cvsheader"
                  . "<p><b>Check whether the directory $cvsroot/CVSROOT exists "
                  . "and the script has write-access to the CVSROOT/history "
                  . "file if it exists."
                  . "<br>The script needs to place lock files in the "
                  . "directory the file is in as well.</b>");
     }      }
     $| = 1;      $| = 1;
   
Line 1640  sub cvswebMarkup($$$) {
Line 1436  sub cvswebMarkup($$$) {
     my $url = download_url($fileurl, $revision, $mimetype);      my $url = download_url($fileurl, $revision, $mimetype);
     print "<HR noshade>";      print "<HR noshade>";
     if ($mimetype =~ /^image/) {      if ($mimetype =~ /^image/) {
         printf '<IMG SRC="%s"><BR>', hrefquote("$url$barequery");          printf '<IMG SRC="%s"><BR>', htmlquote("$url$barequery");
     }      }
     elsif ($mimetype =~ m%^application/pdf%) {      elsif ($mimetype =~ m%^application/pdf%) {
         printf '<EMBED SRC="%s" WIDTH="100%"><BR>', hrefquote("$url$barequery");          printf '<EMBED SRC="%s" WIDTH="100%"><BR>', htmlquote("$url$barequery");
     }      }
     else {      else {
         print "<PRE>";          print "<PRE>";
Line 1674  sub viewable($) {
Line 1470  sub viewable($) {
 sub doDiff($$$$$$) {  sub doDiff($$$$$$) {
         my($fullname, $r1, $tr1, $r2, $tr2, $f) = @_;          my($fullname, $r1, $tr1, $r2, $tr2, $f) = @_;
         my $fh = do {local(*FH);};          my $fh = do {local(*FH);};
         my ($rev1, $rev2, $sym1, $sym2, $f1, $f2);          my ($rev1, $rev2, $sym1, $sym2, @difftype, $diffname, $f1, $f2);
   
         if ($r1 =~ /([^:]+)(:(.+))?/) {          if ($r1 =~ /([^:]+)(:(.+))?/) {
             $rev1 = $1;              $rev1 = $1;
Line 1708  sub doDiff($$$$$$) {
Line 1504  sub doDiff($$$$$$) {
             ($rev1, $sym1) = ($rev2, $sym2);              ($rev1, $sym1) = ($rev2, $sym2);
             ($rev2, $sym2) = ($tmp1, $tmp2);              ($rev2, $sym2) = ($tmp1, $tmp2);
         }          }
         my $difftype = $DIFFTYPES{$f};          my $human_readable = 0;
           if ($f eq 'c') {
         if (!$difftype) {              @difftype = qw{-c};
               $diffname = "Context diff";
           }
           elsif ($f eq 's') {
               @difftype = qw{--side-by-side --width=164};
               $diffname = "Side by Side";
           }
           elsif ($f eq 'H') {
               $human_readable = 1;
               @difftype = qw{--unified=15};
               $diffname = "Long Human readable";
           }
           elsif ($f eq 'h') {
               @difftype =qw{-u};
               $human_readable = 1;
               $diffname = "Human readable";
           }
           elsif ($f eq 'u') {
               @difftype = qw{-u};
               $diffname = "Unidiff";
           }
           else {
             fatal ("400 Bad arguments", "Diff format $f not understood");              fatal ("400 Bad arguments", "Diff format $f not understood");
         }          }
   
         my @difftype       = @{$difftype->{'opts'}};  
         my $human_readable = $difftype->{'colored'};  
   
         # apply special options          # apply special options
         if ($showfunc) {          if ($showfunc) {
             push @difftype, '-p' if $f ne 's';              push @difftype, '-p' if $f =~ /^[cHhu]$/;
   
             my($re1, $re2);              my($re1, $re2);
   
Line 1781  sub doDiff($$$$$$) {
Line 1595  sub doDiff($$$$$$) {
                 s|$cvsroot/||o;                  s|$cvsroot/||o;
                 if ($sym1) {                  if ($sym1) {
                     chop;                      chop;
                     $_ .= " $sym1\n";                      $_ .= " " . $sym1 . "\n";
                 }                  }
             }              }
             elsif (m|^$f2 $cvsroot|o) {              elsif (m|^$f2 $cvsroot|o) {
                 s|$cvsroot/||o;                  s|$cvsroot/||o;
                 if ($sym2) {                  if ($sym2) {
                     chop;                      chop;
                     $_ .= " $sym2\n";                      $_ .= " " . $sym2 . "\n";
                 }                  }
             }              }
             print $_;              print $_;
Line 1856  sub getDirLogs($$@) {
Line 1670  sub getDirLogs($$@) {
 again:  again:
         if ($state eq "head") {          if ($state eq "head") {
             #$rcsfile = $1 if (/^RCS file: (.+)$/); #not used (yet)              #$rcsfile = $1 if (/^RCS file: (.+)$/); #not used (yet)
               $filename = $1 if (/^Working file: (.+)$/);
             if (/^Working file: (.+)$/) {              $head = $1 if (/^head: (.+)$/);
                 $filename = $1;              $branch = $1 if (/^branch: (.+)$/);
             } elsif (/^head: (.+)$/) {          }
                 $head = $1;          if ($state eq "head" && /^symbolic names/) {
             } elsif (/^branch: (.+)$/) {              $state = "tags";
                 $branch = $1              ($branch = $head) =~ s/\.\d+$// if (!defined($branch));
             } elsif (/^symbolic names:/) {              $branch =~ s/(\.?)(\d+)$/${1}0.$2/;
                 $state = "tags";              $symrev{MAIN} = $branch;
                 ($branch = $head) =~ s/\.\d+$// if (!defined($branch));              $symrev{HEAD} = $branch;
                 $branch =~ s/(\d+)$/0.$1/;              $alltags{MAIN} = 1;
                 $symrev{MAIN} = $branch;              $alltags{HEAD} = 1;
                 $symrev{HEAD} = $branch;              push (@filetags, "MAIN", "HEAD");
                 $alltags{MAIN} = 1;  
                 $alltags{HEAD} = 1;  
                 push (@filetags, "MAIN", "HEAD");  
             } elsif (/$LOG_REVSEPARATOR/o) {  
                 $state = "log";  
                 $rev = undef;  
                 $date = undef;  
                 $log = "";  
                 # Try to reconstruct the relative filename if RCS spits out a full path  
                 $filename =~ s%^\Q$DirName\E/%%;  
             }  
             next;              next;
         }          }
         if ($state eq "tags") {          if ($state eq "tags" &&
             if (/^\s+(.+):\s+([\d\.]+)\s+$/) {                              /^\s+(.+):\s+([\d\.]+)\s+$/) {
                 push (@filetags, $1);              push (@filetags, $1);
                 $symrev{$1} = $2;              $symrev{$1} = $2;
                 $alltags{$1} = 1;              $alltags{$1} = 1;
               next;
           }
           if ($state eq "tags" && /^\S/) {
               if (defined($tag) && (defined($symrev{$tag}) || $tag eq "HEAD")) {
                   $revwanted = $tag eq "HEAD" ? $symrev{"MAIN"} : $symrev{$tag};
                   ($branch = $revwanted) =~ s/\b0\.//;
                   ($branchpoint = $branch) =~ s/\.?\d+$//;
                   $revwanted = undef if ($revwanted ne $branch);
               }
               elsif (defined($tag) && $tag ne "HEAD") {
                   print "Tag not found, skip this file" if ($verbose);
                   $state = "skip";
                 next;                  next;
             } elsif (/^\S/) {  
                 if (defined($tag)) {  
                     if(defined($symrev{$tag}) || $tag eq "HEAD") {  
                         $revwanted = $symrev{$tag eq "HEAD" ? "MAIN" : $tag};  
                         ($branch = $revwanted) =~ s/\b0\.//;  
                         ($branchpoint = $branch) =~ s/\.?\d+$//;  
                         $revwanted = undef if ($revwanted ne $branch);  
                     } elsif ($tag ne "HEAD") {  
                         print "Tag not found, skip this file" if ($verbose);  
                         $state = "skip";  
                         next;  
                     }  
                 }  
                 foreach my $tagfound (@filetags) {  
                     $tags{$tagfound} = 1;  
                 }  
                 $state = "head";  
                 goto again;  
             }              }
               foreach my $tagfound (@filetags) {
                   $tags{$tagfound} = 1;
               }
               $state = "head";
               goto again;
         }          }
           if ($state eq "head" && /$LOG_REVSEPARATOR/o) {
               $state = "log";
               $rev = undef;
               $date = undef;
               $log = "";
               # Try to reconstruct the relative filename if RCS spits out a full path
               $filename =~ s%^\Q$DirName\E/%%;
               next;
           }
         if ($state eq "log") {          if ($state eq "log") {
             if (/$LOG_REVSEPARATOR/o || /$LOG_FILESEPARATOR/o) {              if (/$LOG_REVSEPARATOR/o || /$LOG_FILESEPARATOR/o) {
                 # End of a log entry.                  # End of a log entry.
                 my $revbranch;                  my $revbranch;
                 ($revbranch = $rev) =~ s/\.\d+$//;                  ($revbranch = $rev) =~ s/\.\d+$//;
                 print "$filename $rev Wanted: $revwanted ",                  print "$filename $rev Wanted: $revwanted "
                   "Revbranch: $revbranch Branch: $branch ",                      . "Revbranch: $revbranch Branch: $branch "
                     "Branchpoint: $branchpoint\n" if ($verbose);                      . "Branchpoint: $branchpoint\n" if ($verbose);
                 if (!defined($revwanted) && defined($branch)                  if (!defined($revwanted) && defined($branch)
                     && $branch eq $revbranch || !defined($tag)) {                      && $branch eq $revbranch || !defined($tag)) {
                     print "File revision $rev found for branch $branch\n"                      print "File revision $rev found for branch $branch\n"
Line 1953  again:
Line 1764  again:
                 next;                  next;
             }              }
             else {              else {
                 $log .= $_;                  $log = $log . $_;
             }              }
         }          }
         if (/$LOG_FILESEPARATOR/o) {          if (/$LOG_FILESEPARATOR/o) {
Line 2090  sub readLog($;$) {
Line 1901  sub readLog($;$) {
 # is the first commit listed on the appropriate branch.  # is the first commit listed on the appropriate branch.
 # This is not neccesary the same revision as marked as head in the RCS file.  # This is not neccesary the same revision as marked as head in the RCS file.
         my $headrev = $curbranch || "1";          my $headrev = $curbranch || "1";
         ($symrev{"MAIN"} = $headrev) =~ s/(\d+)$/0.$1/;          ($symrev{"MAIN"} = $headrev) =~ s/(\.?)(\d+)$/${1}0.$2/;
           revision:
         foreach $rev (@revorder) {          foreach $rev (@revorder) {
             if ($rev =~ /^(\S*)\.\d+$/ && $headrev eq $1) {              if ($rev =~ /^(\S*)\.\d+$/ && $headrev eq $1) {
                 $symrev{"HEAD"} = $rev;                  $symrev{"HEAD"} = $rev;
                 last;                  last revision;
             }              }
         }          }
         ($symrev{"HEAD"} = $headrev) =~ s/\.\d+$//          ($symrev{"HEAD"} = $headrev) =~ s/\.\d+$//
Line 2125  sub readLog($;$) {
Line 1937  sub readLog($;$) {
                 # with the branch number 0.A, with the exception that                  # with the branch number 0.A, with the exception that
                 # it has no head to translate to if there is nothing on                  # it has no head to translate to if there is nothing on
                 # the branch, but I guess this can never happen?                  # the branch, but I guess this can never happen?
                 #  
                 # (the code below gracefully forgets about the branch                  # (the code below gracefully forgets about the branch
                 # if it should happen)                  # if it should happen)
                 #                  #
Line 2133  sub readLog($;$) {
Line 1944  sub readLog($;$) {
                 $branch = $3;                  $branch = $3;
                 $branchrev = $head . ($head ne "" ? "." : "") . $branch;                  $branchrev = $head . ($head ne "" ? "." : "") . $branch;
                 my $regex;                  my $regex;
                 $regex = quotemeta $branchrev;                  ($regex = $branchrev) =~ s/\./\\./g;
                 $rev = $head;                  $rev = $head;
   
                   revision:
                 foreach my $r (@revorder) {                  foreach my $r (@revorder) {
                     if ($r =~ /^${regex}\b/) {                      if ($r =~ /^${regex}\b/) {
                         $rev = $branchrev;                          $rev = $branchrev;
                         last;                          last revision;
                     }                      }
                 }                  }
                 next if ($rev eq "");                  next if ($rev eq "");
Line 2195  sub readLog($;$) {
Line 2007  sub readLog($;$) {
   
 }  }
   
 sub printDiffLinks($$) {  
     my($text, $url) = @_;  
     my @extra;  
   
     local $_;  
     for ($DIFFTYPES{$defaultDiffType}{'colored'} ? qw(u) : qw(h)) {  
         my $f = $_ eq $defaultDiffType ? '' : $_;  
   
         push @extra, &link(lc $DIFFTYPES{$_}{'descr'}, "$url&f=$f");  
     }  
   
     print &link($text, $url), ' (', join(', ', @extra), ')';  
 }  
   
 sub printLog($;$) {  sub printLog($;$) {
         my ($link, $br, $brp);          my ($link, $br, $brp);
         ($_,$link) = @_;          ($_,$link) = @_;
Line 2285  sub printLog($;$) {
Line 2083  sub printLog($;$) {
         }          }
         if (defined @mytz) {          if (defined @mytz) {
             my ($est) = $mytz[(localtime($date{$_}))[8]];              my ($est) = $mytz[(localtime($date{$_}))[8]];
             print ", <i>", scalar localtime($date{$_}), " $est</i> (";              print ", <i>" . scalar localtime($date{$_}) . " $est</i> (";
         } else {          } else {
             print ", <i>", scalar gmtime($date{$_}), " UTC</i> (";              print ", <i>" . scalar gmtime($date{$_}) . " UTC</i> (";
         }          }
         print readableTime(time() - $date{$_},1), " ago)";          print readableTime(time() - $date{$_},1) . " ago)";
         print " by ";          print " by ";
         print "<i>", $author{$_}, "</i>\n";          print "<i>" . $author{$_} . "</i>\n";
         print "<BR>Branch: <b>",$link?link_tags($revsym{$br}):$revsym{$br},"</b>\n"          print "<BR>Branch: <b>",$link?link_tags($revsym{$br}):$revsym{$br},"</b>\n"
             if ($revsym{$br});              if ($revsym{$br});
         print "<BR>CVS Tags: <b>",$link?link_tags($revsym{$_}):$revsym{$_},"</b>"          print "<BR>CVS Tags: <b>",$link?link_tags($revsym{$_}):$revsym{$_},"</b>"
Line 2333  sub printLog($;$) {
Line 2131  sub printLog($;$) {
                                   $barequery);                                    $barequery);
   
                 print " to previous ";                  print " to previous ";
                 printDiffLinks($prev, $url);                  print &link($prev, $url);
                   if (!$hr_default) { # offer a human readable version if not default
                       print &link('colored', "$url&f=h");
                   }
             }              }
             #              #
             # Plus, if it's on a branch, and it's not a vendor branch,              # Plus, if it's on a branch, and it's not a vendor branch,
Line 2346  sub printLog($;$) {
Line 2147  sub printLog($;$) {
                                   $barequery);                                    $barequery);
   
                 print " to branchpoint ";                  print " to branchpoint ";
                 printDiffLinks($brp, $url);                  print &link($brp, $url);
                   if (!$hr_default) { # offer a human readable version if not default
                       print &link('colored', "$url&f=h");
                   }
             }              }
             #              #
             # Plus, if it's on a branch, and it's not a vendor branch,              # Plus, if it's on a branch, and it's not a vendor branch,
Line 2379  sub printLog($;$) {
Line 2183  sub printLog($;$) {
                                       $barequery);                                        $barequery);
   
                     print " next main ";                      print " next main ";
                     printDiffLinks($nextmain, $url);                      print &link($nextmain, $url);
                       if (!$hr_default) { # offer a human readable version if not default
                           print &link('colored', "$url&f=h");
                       }
                 }                  }
             }              }
             # Plus if user has selected only r1, then present a link              # Plus if user has selected only r1, then present a link
Line 2394  sub printLog($;$) {
Line 2201  sub printLog($;$) {
                                   $barequery);                                    $barequery);
   
                 print " to selected ";                  print " to selected ";
                 printDiffLinks($input{'r1'}, $url);                  print &link($input{'r1'}, $url);
                   if (!$hr_default) { # offer a human readable version if not default
                       print &link('colored', "$url&f=h");
                   }
             }              }
         }          }
         print "<PRE>\n";          print "<PRE>\n";
Line 2474  sub doLog($) {
Line 2284  sub doLog($) {
         print "<HR noshade>\n";          print "<HR noshade>\n";
         print "<TABLE>";          print "<TABLE>";
         print "<FORM METHOD=\"GET\" ACTION=\"$scriptwhere\">\n";          print "<FORM METHOD=\"GET\" ACTION=\"$scriptwhere\">\n";
         print "<TR><TD align=right>Preferred Diff type:</TD>";          print "<TR><TD align=right>Preferred Diff type:</TD>";
         print "<TD>";          print "<TD>";
         printDiffSelect($use_java_script);          printDiffSelect($use_java_script);
         print "</TD><TD></TD></TR>\n";          print "</TD><TD></TD></TR>\n";
         if (@branchnames) {          if (@branchnames) {
Line 2510  sub doLog($) {
Line 2320  sub doLog($) {
         print "<TR><TD align=right>";          print "<TR><TD align=right>";
         print "<A name=logsort></A>\n";          print "<A name=logsort></A>\n";
         print "Sort log by:</TD>";          print "Sort log by:</TD>";
         print "<TD>";          print "<TD><SELECT NAME=\"logsort\"";
         printLogSortSelect($use_java_script);          print " onchange=\"submit()\"" if ($use_java_script);
         print "</TD>";          print ">\n";
           print "<OPTION VALUE=cvs",$logsort eq "cvs" ? " SELECTED" : "", ">Not sorted";
           print "<OPTION VALUE=date",$logsort eq "date" ? " SELECTED" : "", ">Commit date";
           print "<OPTION VALUE=rev",$logsort eq "rev" ? " SELECTED" : "", ">Revision";
           print "</SELECT></TD>";
         print "<TD><INPUT TYPE=SUBMIT VALUE=\"  Set  \"></TD>";          print "<TD><INPUT TYPE=SUBMIT VALUE=\"  Set  \"></TD>";
         print "</FORM>\n";          print "</FORM>\n";
         print "</TR></TABLE>";          print "</TR></TABLE>";
Line 2582  sub human_readable_diff($){
Line 2396  sub human_readable_diff($){
     $date2 = $r2d;      $date2 = $r2d;
   }    }
   
   print "<h3 align=center>Diff for /$where_nd between version $rev1 and $rev2</h3>\n",    print "<h3 align=center>Diff for /$where_nd between version $rev1 and $rev2</h3>\n";
     "<table border=0 cellspacing=0 cellpadding=0 width=\"100%\">\n",  
       "<tr bgcolor=\"#ffffff\">\n",    print "<table border=0 cellspacing=0 cellpadding=0 width=\"100%\">\n";
         "<th width=\"50%\" valign=TOP>",    print "<tr bgcolor=\"#ffffff\">\n";
           "version $rev1";    print "<th width=\"50%\" valign=TOP>";
     print "version $rev1";
   print ", $date1" if (defined($date1));    print ", $date1" if (defined($date1));
   print "<br>Tag: $sym1\n" if ($sym1);    print "<br>Tag: $sym1\n" if ($sym1);
   print "</th>\n",    print "</th>\n";
     "<th width=\"50%\" valign=TOP>",    print "<th width=\"50%\" valign=TOP>";
       "version $rev2";    print "version $rev2";
   print ", $date2" if (defined($date2));    print ", $date2" if (defined($date2));
   print "<br>Tag: $sym2\n" if ($sym1);    print "<br>Tag: $sym2\n" if ($sym1);
   print "</th>\n";    print "</th>\n";
Line 2714  sub navigateHeader($$$$$) {
Line 2529  sub navigateHeader($$$$$) {
     my ($swhere,$path,$filename,$rev,$title) = @_;      my ($swhere,$path,$filename,$rev,$title) = @_;
     $swhere = "" if ($swhere eq $scriptwhere);      $swhere = "" if ($swhere eq $scriptwhere);
     $swhere = urlencode($filename) if ($swhere eq "");      $swhere = urlencode($filename) if ($swhere eq "");
     print qq`<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">`;      print "<\!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\">";
     print "<HTML>\n<HEAD>\n";      print "<HTML>\n<HEAD>\n";
     print qq`<META name="robots" content="nofollow">\n`;      print '<!-- CVSweb $zRevision: 1.103 $  $Revision$ -->';
     print '<!-- CVSweb $zRevision: 1.104 $  $kRevision: 1.47 $ -->';  
     print "\n<TITLE>$path$filename - $title - $rev</TITLE></HEAD>\n";      print "\n<TITLE>$path$filename - $title - $rev</TITLE></HEAD>\n";
     print  "$body_tag_for_src\n";      print  "$body_tag_for_src\n";
     print "<table width=\"100%\" border=0 cellspacing=0 cellpadding=1 bgcolor=\"$navigationHeaderColor\">";      print "<table width=\"100%\" border=0 cellspacing=0 cellpadding=1 bgcolor=\"$navigationHeaderColor\">";
Line 2733  sub navigateHeader($$$$$) {
Line 2547  sub navigateHeader($$$$$) {
 sub plural_write($$) {  sub plural_write($$) {
     my ($num,$text) = @_;      my ($num,$text) = @_;
     if ($num != 1) {      if ($num != 1) {
         $text .= "s";          $text = $text . "s";
     }      }
     if ($num > 0) {      if ($num > 0) {
         return join(' ', $num, $text);          return $num . " " . $text;
     }      }
     else {      else {
         return "";          return "";
Line 2780  sub readableTime($$) {
Line 2594  sub readableTime($$) {
         my $resttime = plural_write(int ($rest / $break),          my $resttime = plural_write(int ($rest / $break),
                                 $desc{$break});                                  $desc{$break});
         if ($resttime) {          if ($resttime) {
             $retval .= ", $resttime";              $retval = $retval . ", " . $resttime;
         }          }
     }      }
   
Line 2810  sub clickablePath($$) {
Line 2624  sub clickablePath($$) {
         my $wherepath = '';          my $wherepath = '';
         my ($lastslash) = $pathname =~ m|/$|;          my ($lastslash) = $pathname =~ m|/$|;
         foreach (split(/\//, $pathname)) {          foreach (split(/\//, $pathname)) {
             $retval .= " / ";              $retval = $retval . " / ";
             $wherepath .= "/$_";              $wherepath = $wherepath . '/' . $_;
             my ($last) = "$wherepath/" eq "/$pathname"              my ($last) = "$wherepath/" eq "/$pathname"
                 || $wherepath eq "/$pathname";                  || $wherepath eq "/$pathname";
             if ($clickLast || !$last) {              if ($clickLast || !$last) {
Line 2855  sub chooseCVSRoot() {
Line 2669  sub chooseCVSRoot() {
         foreach $k (@foo) {          foreach $k (@foo) {
             print "<option value=\"$k\"";              print "<option value=\"$k\"";
             print " selected" if ($k eq $cvstree);              print " selected" if ($k eq $cvstree);
             print ">", ($CVSROOTdescr{$k} ? $CVSROOTdescr{$k} : $k), "</option>\n";              print ">" . ($CVSROOTdescr{$k} ? $CVSROOTdescr{$k} :
                           $k). "</option>\n";
         }          }
         print "</select>\n</td>";          print "</select>\n</td>";
         print "<td><input type=submit value=\"Go\"></td>";          print "<td><input type=submit value=\"Go\"></td>";
Line 2940  sub download_link($$$;$) {
Line 2755  sub download_link($$$;$) {
     my ($url, $revision, $textlink, $mimetype) = @_;      my ($url, $revision, $textlink, $mimetype) = @_;
     my ($fullurl) = download_url($url, $revision, $mimetype);      my ($fullurl) = download_url($url, $revision, $mimetype);
   
     printf '<A HREF="%s"', hrefquote("$fullurl$barequery");      printf '<A HREF="%s"', htmlquote("$fullurl$barequery");
   
     if ($open_extern_window && (!defined($mimetype) || $mimetype ne "text/x-cvsweb-markup")) {      if ($open_extern_window && (!defined($mimetype) || $mimetype ne "text/x-cvsweb-markup")) {
         print ' target="cvs_checkout"';          print ' target="cvs_checkout"';
Line 2974  sub download_link($$$;$) {
Line 2789  sub download_link($$$;$) {
               if (defined($extern_window_height));                if (defined($extern_window_height));
   
             printf q` onClick="window.open('%s','cvs_checkout','%s');"`,              printf q` onClick="window.open('%s','cvs_checkout','%s');"`,
               hrefquote($fullurl), join(',', @attr);                htmlquote($fullurl), join(',', @attr);
         }          }
     }      }
     print "><b>$textlink</b></A>";      print "><b>$textlink</b></A>";
Line 3014  sub urlencode($) {
Line 2829  sub urlencode($) {
   
     s/[\000-+{-\377]/sprintf("%%%02x", ord($&))/ge;      s/[\000-+{-\377]/sprintf("%%%02x", ord($&))/ge;
   
     $_;  
          $_;
 }  }
   
 sub htmlquote($) {  sub htmlquote($) {
Line 3029  sub htmlquote($) {
Line 2845  sub htmlquote($) {
     $_;      $_;
 }  }
   
 sub htmlunquote($) {  
     local($_) = @_;  
   
     # Special Characters; RFC 1866  
     s/&quot;/\"/g;  
     s/&lt;/</g;  
     s/&gt;/>/g;  
     s/&amp;/&/g;  
   
     $_;  
 }  
   
 sub hrefquote($) {  
     local($_) = @_;  
   
     y/ /+/;  
   
     htmlquote($_)  
 }  
   
 sub http_header(;$) {  sub http_header(;$) {
     my $content_type = shift || "text/html";      my $content_type = shift || "text/html";
     if (defined($moddate)) {      if (defined($moddate)) {
Line 3056  sub http_header(;$) {
Line 2852  sub http_header(;$) {
             Apache->request->header_out("Last-Modified" => scalar gmtime($moddate) . " GMT");              Apache->request->header_out("Last-Modified" => scalar gmtime($moddate) . " GMT");
         }          }
         else {          else {
             print "Last-Modified: ", scalar gmtime($moddate), " GMT\r\n";              print "Last-Modified: " . scalar gmtime($moddate) . " GMT\r\n";
         }          }
     }      }
     if ($is_mod_perl) {      if ($is_mod_perl) {
Line 3083  sub http_header(;$) {
Line 2879  sub http_header(;$) {
             }              }
             select(GZIP);              select(GZIP);
             $gzip_open = 1;              $gzip_open = 1;
 #           print "<!-- gzipped -->" if ($content_type =~ m|^text/html\b|);  #           print "<!-- gzipped -->" if ($content_type eq "text/html");
         }          }
         else {          else {
             if ($is_mod_perl) {              if ($is_mod_perl) {
Line 3107  sub http_header(;$) {
Line 2903  sub http_header(;$) {
   
 sub html_header($) {  sub html_header($) {
     my ($title) = @_;      my ($title) = @_;
     my $version = '$zRevision: 1.104 $  $kRevision: 1.47 $'; #'      my $version = '$zRevision: 1.103 $  $Revision$'; #'
     http_header(defined($charset) ? "text/html; charset=$charset" : "text/html");      http_header();
     print <<EOH;      print <<EOH;
 <!doctype html public "-//W3C//DTD HTML 4.0 Transitional//EN"  <!doctype html public "-//W3C//DTD HTML 4.0 Transitional//EN"
  "http://www.w3.org/TR/REC-html40/loose.dtd">   "http://www.w3.org/TR/REC-html40/loose.dtd">
 <html>  <html>
 <head>  
 <meta name="robots" content="nofollow">  
 <title>$title</title>  <title>$title</title>
 <!-- CVSweb $version -->  <!-- CVSweb $version -->
 </head>  </head>

Legend:
Removed from v.1.1.1.16  
changed lines
  Added in v.3.35

CVSweb