Diff for /loncom/build/system_dependencies/perltest.pl between versions 1.1 and 1.10

version 1.1, 2001/09/30 20:57:58 version 1.10, 2003/02/03 18:03:52
Line 1 Line 1
 #!/usr/bin/perl  #!/usr/bin/perl
   
 # The LearningOnline Network with CAPA  # perltest.pl - script to test the status of perl modules on a LON-CAPA system
 #  #
 # Testing to see if all needed perl components are on  # $Id$
 # a system  
 #  #
 # YEAR=2001  ###
 # 9/30 Scott Harrison  
   =pod
   
   =head1 NAME
   
   B<perltest.pl> - Test status of perl modules installed on a LON-CAPA system.
   
   =cut
   
   # Written to help LON-CAPA (The LearningOnline Network with CAPA)
 #  #
 # I wish there were a more straightforward way to do this.  But I can't  
 # seem to find the right internals in the perl programming language.  =pod
 # Also, this program is only a "modest" effort to LOOK and see whether  
   =head1 SYNOPSIS
   
   perl perltest.pl [MODE]
   
   This script is located inside the LON-CAPA source code tree.
   This script is invoked by test-related targets inside
   F<loncapa/loncom/build/Makefile>.
   
   This script is also used as a CGI script and is installed
   at the file location of F</home/httpd/cgi-bin/perltest.pl>.
   
   MODE, when left blank, the output defaults to 'statusreport' mode.
   Except however, if $ENV{'QUERY_STRING'} exists, in which case
   'html' mode is safely assumed.
   
   Here is a complete list of MODEs.
   
   =over 4
   
   =item html
   
   A web page detailing the status of CPAN distributions on a LON-CAPA server
   (as well as methods for resolution).
   
   =item synopsis
   
   Plain-text output which just summarizes the status of
   expected CPAN distributions on a system.  (This is what a
   user sees when running the ./TEST command.)
   
   =item statusreport
   
   Plain-text output which provides a detailed status report of
   CPAN distributions on a LON-CAPA server (as well as methods
   for resolution).
   
   =back
   
   =head1 DESCRIPTION
   
   This program tests the status of perl modules installed on a LON-CAPA system.
   As with the other LON-CAPA test scripts, when reasonable, I try
   to avoid importing functionality from other LON-CAPA modules so as to
   avoid indirectly testing software dependencies.
   
   =head2 ORGANIZATION OF THIS PERL SCRIPT
   
   The script is organized into the following sections.
   
   =over 4
   
   =item 1.
   
   Process version information of this file.
   
   =item 2.
   
   Determine output mode for the script.
   
   =item 3.
   
   Output header information.
   
   =item 4.
   
   Make sure the perl version is suitably high.
   
   =item 5.
   
   Make sure we have the find command.
   
   =item 6.
   
   Scan for all the perl modules present on the filesystem.
   
   =item 7.
   
   Read in cpan_distributions.txt.
   
   =item 8.
   
   Loop through all of the needed CPAN distributions and probe the system.
   
   =item 9
   
   Output a report (dependent on output mode).
   
   =item 10
   
   Subroutines.
   
   B<vers_cmp> - compare two version numbers and see which is greater.
   
   B<have_vers> - syntax check the version number and call B<vers_cmp>.
   
   =back
   
   =head1 STATUS
   
   Ratings: 1=horrible 2=poor 3=fair 4=good 5=excellent
   
   =over 4
   
   =item Organization
   
   5
   
   =item Functionality
   
   5
   
   =item Has it been tested?
   
   4
   
   =back
   
   =head1 AUTHOR
   
   This software is distributed under the General Public License,
   version 2, June 1991 (which is the same terms as LON-CAPA).
   
   This is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.
   
   This software is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.
   
   You should have received a copy of the GNU General Public License
   along with this software; if not, write to the Free Software
   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
   
   =cut
   
   # =================================== Process version information of this file.
   my $VERSION = sprintf("%d.%02d", q$Revision$ =~ /(\d+)\.(\d+)/);
   
   # ========================== Determine the mode that this script should run in.
   my $mode;
   $mode=shift(@ARGV) if @ARGV;
   unless ( $mode )
     {
       $mode = 'statusreport';
     }
   if ( defined($ENV{'QUERY_STRING'}) )
     {
       $mode = 'html';
     }
   
   # ================================================== Output header information.
   my $hostname = `hostname`; chomp($hostname);
   my $date = `date`; chomp($date);
   
   # --- html mode blurb
   if ($mode eq "html") {
       print(<<END);
   Content-type: text/html
   
   <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
    "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
   <html>
   <head>
   <meta http-equiv="Content-Type" content="text/html; charset=utf-8"></meta>
   <title>CPAN perl status report; $hostname; $date</title>
   </head>
   <body bgcolor="white">
   <h1>CPAN perl status report</h1>
   <pre>
   END
   }
   
   print('Running perltest.pl, version '.$VERSION.'.'."\n");
   print('(Test status of perl modules installed on a LON-CAPA system).'."\n");
   
   # This program is only a "modest" effort to LOOK and see whether
 # necessary perl system dependencies are present.  I do not yet  # necessary perl system dependencies are present.  I do not yet
 # try to actually run tests against each needed perl module.  # try to actually run tests against each needed perl module.
   # Eventually, all modules will be version-checked, and reasonable
   # testing implemented.
   
 # --------------------------------- Make sure the perl version is suitably high  # ================================ Make sure the perl version is suitably high.
 my $ret=system("find --version 1>/dev/null");  print('Checking version of perl'."\n");
 if ($]<5.005) {  print(`perl --version`);
     die "DEPENDENCY FAILURE: LON-CAPA requires perl version >= 5.005.\n";  unless (eval("require 5.005"))
 }    {
       die('**** ERROR **** DEPENDENCY FAILURE: require perl version >= 5.005.'.
    "\n".'Do you even have perl installed on your system?'."\n");
     }
   else
     {
       print('Perl >= 5.005...okay'."\n");
     }
   
 # ------------------------------------------ Make sure we have the find command  # ========================================= Make sure we have the find command.
 my $ret=system("find --version 1>/dev/null");  my $ret = system("find --version 1>/dev/null");
 if ($ret) {  if ($ret)
     die "DEPENDENCY FAILURE: A test program requires the GNU ".    {
  "'find' utility.\n";      die('**** ERROR **** DEPENDENCY FAILURE: perltest.pl requires the GNU '.
 }   "'find'".' utility.'."\n");
     }
   else
     {
       print('find command exists...okay'."\n");
     }
   
 # ----------------------------------------- Read in all the system perl modules  # ==================== Scan for all the perl modules present on the filesystem.
 print "Scanning for perl modules...\n";  print('Scanning for perl modules...'."\n");
 my $modules;   my $big_module_string; # All the modules glued together in a string.
 foreach my $inc (@INC) {  my $number_of_modules = 0; # The total number of modules available in system.
     my @m=`find $inc -type f -name '*.pm'`;  # --- Build a pattern matching string.
     foreach my $mod (@m) {  foreach my $inc (@INC)
  $modules.=$mod;    {
     }      my @m = `find $inc -maxdepth 2000 -type f -name '*.pm'`;
 }      foreach my $module (@m)
         {
    $big_module_string .= $module;
    $number_of_modules++;
         }
     }
   # --- Notify user of the number of modules.
   print('There are '.$number_of_modules.
         ' perl modules present on your filesystem.'."\n");
   
 # --------------- Seeing if dependencies are in our "full" list of perl modules  my %dist_module_hash; # Relate the distributions to their VersionFrom modules.
 my %dependencies=qw(  my %module_name_on_filesystem; # Relate module name to filesystem syntax.
     GD/Barcode.pm 1  my %dist_dev_version_hash; # Expected development version of CPAN distribution.
     GD/Barcode/COOP2of5.pm 1  my %dist_stable_version_hash; # Expected stable version of CPAN distribution.
     GD/Barcode/Code39.pm 1  my %module_dev_version_hash; # development version of versionfrom_module.
     GD/Barcode/EAN13.pm 1  my %module_stable_version_hash; # stable version of versionfrom_module.
     GD/Barcode/EAN8.pm 1  
     GD/Barcode/IATA2of5.pm 1  # ============================================= Read in cpan_distributions.txt.
     GD/Barcode/ITF.pm 1  
     GD/Barcode/Industrial2of5.pm 1  # A brief description of CPAN (Comprehensive Perl Archive Network):
     GD/Barcode/Matrix2of5.pm 1  # CPAN software is not released as separate perl modules.
     GD/Barcode/NW7.pm 1  # CPAN software is released as "distributions" (also called "dists").
     GD/Barcode/UPCA.pm 1  # Each distribution consists of multiple perl modules.
     GD/Barcode/UPCE.pm 1  # For instance, the dist HTML-Tree (http://search.cpan.org/dist/HTML-Tree/)
     IO/Dir.pm 1  # consists of the modules HTML::AsSubs, HTML::Element, HTML::Element::traverse,
     IO/Poll.pm 1  # HTML::Parse, HTML::TreeBuilder, and HTML::Tree.
     IO/Socket/INET.pm 1  # Most (but not all) distributions have versions which are defined
     IO/Socket/UNIX.pm 1  # by one of their modules.  For the syntax of cpan_distributions.txt,
     Algorithm/Diff.pm 1  # please read the comments inside cpan_distributions.txt.
     Algorithm/DiffOld.pm 1  
     Bundle/LWP.pm 1  # Open cpan_distributions.txt.
     File/Listing.pm 1  open(IN,'<cpan_distributions.txt') or
     HTML/AsSubs.pm 1      die('**** ERROR **** Cannot find cpan_distributions.txt'."\n");
     HTML/Element.pm 1  
     HTML/Element/traverse.pm 1  while(<IN>) # Loop through the lines.
     HTML/Form.pm 1    {
     HTML/Parse.pm 1      next if /^\#/; # Ignore commented lines.
     HTML/Tagset.pm 1      next unless /\S/; # Ignore blank lines.
     HTML/Tree.pm 1  
     HTML/TreeBuilder.pm 1      chomp; # Get rid of the newline at the end of the line.
     HTTP/Cookies.pm 1  
     HTTP/Daemon.pm 1      # Parse the line.
     HTTP/Date.pm 1      my ($dist_name,$dist_dev_version,$dist_stable_version,$versionfrom_info) =
     HTTP/Headers.pm 1   split(/\s+/); # Parse apart the line fields.
     HTTP/Headers/Auth.pm 1      $versionfrom_info =~ /^(.*)\((.*)\)$/; # Parse apart the versionfrom info.
     HTTP/Headers/ETag.pm 1      my ($version_module,$version_match) = ($1,$2); # Parse vals into variables.
     HTTP/Headers/Util.pm 1  
     HTTP/Message.pm 1      # Calculate DevVersion and StableVersion for the VersionFrom module.
     HTTP/Negotiate.pm 1      my $module_dev_version;
     HTTP/Request.pm 1      my $module_stable_version;
     HTTP/Request/Common.pm 1      if ($version_match eq "*") # There is a dist=module version relationship.
     HTTP/Response.pm 1        {
     HTTP/Status.pm 1   $module_dev_version = $dist_dev_version; # module=dist.
     LWP.pm 1   $module_stable_version = $dist_stable_version; # module=dist.
     LWP/Authen/Basic.pm 1        }
     LWP/Authen/Digest.pm 1      else # There is not a dist=module version relationship.
     LWP/Debug.pm 1        {
     LWP/MediaTypes.pm 1   ($module_dev_version,$module_stable_version) = 
     LWP/MemberMixin.pm 1      split(/\,/,$version_match); # module set to customized settings.
     LWP/Protocol.pm 1        }
     LWP/Protocol/data.pm 1  
     LWP/Protocol/file.pm 1      $dist_module_hash{$dist_name} = $version_module; # The big dist index.
     LWP/Protocol/ftp.pm 1  
     LWP/Protocol/gopher.pm 1      # What the module "looks like" on the filesystem.
     LWP/Protocol/http.pm 1      my $version_modulefs = $version_module;
     LWP/Protocol/https.pm 1      $version_modulefs =~ s!::!/!g; $version_modulefs.='.pm';
     LWP/Protocol/mailto.pm 1      $modulefs_hash{$version_module} = $version_modulefs;
     LWP/Protocol/nntp.pm 1  
     LWP/RobotUA.pm 1      # Indexing the expected versions.
     LWP/Simple.pm 1      $module_dev_version_hash{$version_module} = $module_dev_version;
     LWP/UserAgent.pm 1      $module_stable_version_hash{$version_module} = $module_stable_version;
     MIME/Body.pm 1      $dist_dev_version_hash{$dist_name} = $dist_dev_version;
     MIME/Decoder.pm 1      $dist_stable_version_hash{$dist_name} = $dist_stable_version;
     MIME/Decoder/Base64.pm 1    }
     MIME/Decoder/Binary.pm 1  close(IN);
     MIME/Decoder/Gzip64.pm 1  
     MIME/Decoder/NBit.pm 1  # "MISSING"  means that no module is present inside the include path.
     MIME/Decoder/QuotedPrint.pm 1  # "OUTDATED" means that a module is present inside the include path but is
     MIME/Decoder/UU.pm 1  #            an earlier version than expected.
     MIME/Entity.pm 1  # "VERYOKAY" means that the module version is an exact match for the expected
     MIME/Field/ConTraEnc.pm 1  #            version.
     MIME/Field/ContDisp.pm 1  # "OKAY"     means that the module version is more recent than the expected
     MIME/Field/ContType.pm 1  #            version, so things are "probably" okay....  It is still possible
     MIME/Field/ParamVal.pm 1  #            that LON-CAPA is incompatible with the newer distribution version
     MIME/Head.pm 1  #            (corresponding to the module version).
     MIME/Parser.pm 1  my @dev_missing;
     MIME/Parser/Filer.pm 1  my @dev_outdated;
     MIME/Parser/Reader.pm 1  my @dev_okay;
     MIME/Parser/Results.pm 1  my @dev_veryokay;
     MIME/Tools.pm 1  my @stable_missing;
     MIME/Words.pm 1  my @stable_outdated;
     Net/Cmd.pm 1  my @stable_okay;
     Net/Config.pm 1  my @stable_veryokay;
     Net/Domain.pm 1  
     Net/DummyInetd.pm 1  # ===== Loop through all of the needed CPAN distributions and probe the system.
     Net/FTP.pm 1  foreach my $dist (keys %dist_module_hash)
     Net/FTP/A.pm 1    {
     Net/FTP/E.pm 1      my $module = $dist_module_hash{$dist};
     Net/FTP/I.pm 1      my $fs = $modulefs_hash{$module};
     Net/FTP/L.pm 1      my $fsflag = 0;
     Net/FTP/dataconn.pm 1      if ($big_module_string =~ /$fs/)
     Net/NNTP.pm 1        {
     Net/Netrc.pm 1          $fsflag = 1;
     Net/PH.pm 1        }
     Net/POP3.pm 1      my ($vok,$vstr);
     Net/SMTP.pm 1      ($vok,$vstr) = have_vers($module,$module_dev_version_hash{$module});
     Net/SNPP.pm 1      # print "fsflag: $fsflag, vok: $vok, vstr: $vstr, fs: $fs\n";
     Net/Time.pm 1      if ($fsflag and !$vok and $vstr=~/not found/)
     Text/Query.pm 1        {
     Text/Query/Advanced.pm 1   push(@dev_missing,'MISSING  '.$dist.' (want distribution version '.
     Text/Query/Build.pm 1       $dist_dev_version_hash{$dist}.') ?'."\n");
     Text/Query/BuildAdvancedString.pm 1   # The question mark indicates there was a pattern match in the
     Text/Query/BuildSimpleString.pm 1   # big_module_string which would be unexpected.
     Text/Query/Optimize.pm 1   # There is no usual reason to tell the normal LON-CAPA user about this
     Text/Query/Parse.pm 1   # question mark.  This is just source code magic.
     Text/Query/ParseAdvanced.pm 1        }
     Text/Query/ParseSimple.pm 1      elsif (!$fsflag and !$vok and $vstr=~/not found/)
     Text/Query/Simple.pm 1        {
     Text/Query/Solve.pm 1   push(@dev_missing,'MISSING  '.$dist.' (want distribution version '.
     Text/Query/SolveAdvancedString.pm 1       $dist_dev_version_hash{$dist}.')'."\n");
     Text/Query/SolveSimpleString.pm 1        }
     URI.pm 1      elsif ($fsflag and !$vok and $vstr!~/not found/)
     URI/Escape.pm 1        {
     URI/Heuristic.pm 1   push(@dev_outdated,'OUTDATED '.$dist.' wanted module: v'.
     URI/URL.pm 1       $module_dev_version_hash{$module}.'; '.$vstr.' (VERSION_FROM is '.
     URI/WithBase.pm 1       $fs.') want dist version '.$dist_dev_version_hash{$dist}.'.'.
     URI/_foreign.pm 1       "\n");
     URI/_generic.pm 1        }
     URI/_login.pm 1      elsif ($fsflag)
     URI/_query.pm 1        {
     URI/_segment.pm 1   $vstr=~/found v(.*)/;
     URI/_server.pm 1   my $vc=$1;
     URI/_userpass.pm 1   if ($vc eq $module_dev_version_hash{$module})
     URI/data.pm 1            {
     URI/file.pm 1      push(@dev_veryokay,'VERYOKAY '.$dist.' wanted: v'.
     URI/file/Base.pm 1   $module_dev_version_hash{$module}.'; '.$vstr.
     URI/file/FAT.pm 1   ' (VERSION_FROM is '.$fs.') want dist version '.
     URI/file/Mac.pm 1   $dist_dev_version_hash{$dist}."\n");
     URI/file/OS2.pm 1    }
     URI/file/QNX.pm 1   else
     URI/file/Unix.pm 1            {
     URI/file/Win32.pm 1      push(@dev_okay,'OKAY     '.$dist.' wanted: v'.
     URI/ftp.pm 1    $module_dev_version_hash{$module}.'; '.$vstr.
     URI/gopher.pm 1    ' (VERSION_FROM is '.$fs.').'."\n");
     URI/http.pm 1    }
     URI/https.pm 1        }
     URI/ldap.pm 1      ($vok,$vstr) = have_vers($module,$module_stable_version_hash{$module});
     URI/mailto.pm 1      if ($fsflag and !$vok and $vstr=~/not found/)
     URI/news.pm 1        {
     URI/nntp.pm 1   push(@stable_missing,'MISSING  '.$dist.' (want distribution version '.
     URI/pop.pm 1       $dist_stable_version_hash{$dist}.') ?'."\n");
     URI/rlogin.pm 1   # The question mark indicates there was a pattern match in the
     URI/rsync.pm 1   # big_module_string which would be unexpected.
     URI/snews.pm 1   # There is no usual reason to tell the normal LON-CAPA user about this
     URI/telnet.pm 1   # question mark.  This is just source code magic.
     WWW/RobotRules.pm 1        }
     WWW/RobotRules/AnyDBM_File.pm 1      elsif (!$fsflag and !$vok and $vstr=~/not found/)
     Authen/Krb4.pm 1        {
     Bundle/DBD/mysql.pm 1   push(@stable_missing,'MISSING  '.$dist.' (want distribution version '.
     Bundle/DBI.pm 1       $dist_stable_version_hash{$dist}.')'."\n");
     Crypt/DES.pm 1        }
     Crypt/IDEA.pm 1      elsif ($fsflag and !$vok and $vstr!~/not found/)
     DBD/ADO.pm 1        {
     DBD/ExampleP.pm 1   push(@stable_outdated,'OUTDATED '.$dist.' wanted module: v'.
     DBD/Multiplex.pm 1       $module_stable_version_hash{$module}.'; '.$vstr.
     DBD/NullP.pm 1       ' (VERSION_FROM is '.$fs.') want dist version '.
     DBD/Proxy.pm 1       $dist_stable_version_hash{$dist}.'.'."\n");
     DBD/Sponge.pm 1        }
     DBD/mysql.pm 1      elsif ($fsflag)
     DBI.pm 1        {
     DBI/DBD.pm 1   $vstr=~/found v(.*)/;
     DBI/FAQ.pm 1   my $vc=$1;
     DBI/Format.pm 1   if ($vc eq $module_stable_version_hash{$module})
     DBI/ProxyServer.pm 1            {
     DBI/Shell.pm 1      push(@stable_veryokay,'VERYOKAY '.$dist.' wanted: v'.
     DBI/W32ODBC.pm 1   $module_stable_version_hash{$module}.'; '.$vstr.
     Digest.pm 1   ' (VERSION_FROM is '.$fs.') want dist version '.
     Digest/HMAC.pm 1   $dist_stable_version_hash{$dist}."\n");
     Digest/HMAC_MD5.pm 1    }
     Digest/HMAC_SHA1.pm 1   else
     Digest/MD2.pm 1            {
     Digest/MD5.pm 1      push(@stable_okay,'OKAY     '.$dist.' wanted: v'.
     Digest/SHA1.pm 1    $module_stable_version_hash{$module}.'; '.$vstr.
     HTML/Entities.pm 1    ' (VERSION_FROM is '.$fs.').'."\n");
     HTML/Filter.pm 1    }
     HTML/HeadParser.pm 1        }
     HTML/LinkExtor.pm 1    }
     HTML/Parser.pm 1  
     HTML/TokeParser.pm 1  print("\n".'SYNOPSIS'."\n");
     MIME/Base64.pm 1  
     MIME/QuotedPrint.pm 1  # ========================================================== The stable report.
     Math/Cephes.pm 1  print('**** STABLE REPORT (what a production server should worry about)'."\n");
     Math/Cephes/Complex.pm 1  if (@stable_missing)
     Math/Cephes/Fraction.pm 1    {
     Mysql.pm 1      print('There are '.scalar(@stable_missing).' CPAN distributions missing '.
     Mysql/Statement.pm 1    'from this LON-CAPA system.'."\n");
     SHA.pm 1    }
     Safe/Hole.pm 1  else
     Win32/DBIODBC.pm 1    {
     Algorithm/Diff.pm 1      print('All perl modules needed by LON-CAPA appear to be present.'."\n");
     Apache/Constants.pm 1    }
     Apache/File.pm 1  if (@stable_outdated)
     capa.pm 1    {
     CGI.pm 1      print(scalar(@stable_outdated).' CPAN distributions are out-dated '.
     CGI/Cookie.pm 1    'on this LON-CAPA system.'."\n");
     Crypt/DES.pm 1    }
     DBI.pm 1  if (@stable_veryokay)
     Fcntl.pm 1    {
     File/Copy.pm 1      print(scalar(@stable_veryokay).' CPAN distributions are an exact match '.
     FileHandle.pm 1    '(based on version number).'."\n");
     GDBM_File.pm 1  #    print @stable_veryokay;
     HTML/TokeParser.pm 1    }
     HTML/TreeBuilder.pm 1  if (@stable_okay)
     HTTP/Headers.pm 1    {
     IO/Socket.pm 1      print(scalar(@stable_okay).' CPAN dists have a version number '.
     LWP/UserAgent.pm 1    'higher than expected'.
     Math/Cephes.pm 1    ' (probably okay).'. "\n");
     Math/Random.pm 1    }
     Opcode.pm 1  print("\n");
     POSIX.pm 1  
     Safe.pm 1  # ===================================================== The development report.
     Safe/Hole.pm 1  print('**** DEVELOPMENT REPORT (do not worry about this unless you are a'.
     Socket.pm 1        ' coder)'."\n");
     strict.pm 1  if (@dev_missing)
     Text/Query.pm 1    {
     tth.pm 1      print('There are '.scalar(@dev_missing).' CPAN distributions missing '.
     vars.pm 1    'from this LON-CAPA system.'."\n");
     );    }
   else
 my $errorflag=0;    {
 foreach my $dep (keys %dependencies) {      print('All perl modules needed by LON-CAPA appear to be present.'."\n");
     unless ($modules=~/$dep/) {    }
  print "MISSING $dep\n";  if (@dev_outdated)
  $errorflag++;    {
     }      print(scalar(@dev_outdated).' CPAN distributions are out-dated '.
 }    'on this LON-CAPA system.'."\n");
     }
   if (@dev_veryokay)
     {
       print(scalar(@dev_veryokay).' CPAN distributions are an exact match '.
     '(based on version number).'."\n");
   #    print @dev_veryokay;
     }
   if (@dev_okay)
     {
       print(scalar(@stable_okay).' CPAN dists have a version number '.
     'higher than expected'.
     ' (probably okay).'. "\n");
     }
   
   my $detailstream;
   if ($mode eq 'synopsis')
     {
       print("\n".'**** NOTE ****'."\n".
     'After everything completes, please view the CPAN_STATUS_REPORT'.
     ' file for more '."\n".'information on resolving your perl modules.'.
     "\n");
   
       print('* HIT RETURN WHEN READY TO CONTINUE *'."\n");
       my $returnkey=<>;
       open(OUT,'>CPAN_STATUS_REPORT');
       $detailstream=\*OUT;
     }
   else
     {
       $detailstream=\*STDOUT;
     }
   print($detailstream 
         "\n".'DETAILED STATUS REPORT'."\n"); # Header of status report.
   
   # Print advisory notices.
   print($detailstream
         "\n".'(Consult loncapa/doc/otherfiles/perl_modules.txt for '.
         'information on'."\n".
         ' manual build instructions.)'."\n");
   print($detailstream
         "\n".'(**** IMPORTANT NOTICE **** HTML-Parser needs to be patched '.
         "\n".' as described in loncapa/doc/otherfiles/perl_modules.txt)'.
         "\n");
   
   print($detailstream
         "\n".'For manual installation of CPAN distributions, visit'."\n".
         'http://search.cpan.org/dist/DistName'."\n".
         'where DistName is something like "HTML-Parser" or "libwww-perl".'.
         "\n");
   
   print($detailstream
         "\n".'For automatic installation of CPAN distributions, visit'."\n".
         'http://install.lon-capa.org/resources/cpanauto/DistName.bin'."\n".
         'where DistName.bin is something like "HTML-Parser.bin" or '.
         '"libwww-perl.bin".'."\n");
   
   # Print detailed report of stable.
   print($detailstream
         "\n".'STABLE (DETAILED REPORT)'."\n");
   print $detailstream @stable_missing;
   print $detailstream @stable_outdated;
   print $detailstream @stable_veryokay;
   print $detailstream @stable_okay;
   print($detailstream "\n".'DEVELOPMENT (DETAILED REPORT)'."\n");
   print $detailstream @dev_missing;
   print $detailstream @dev_outdated;
   print $detailstream @dev_veryokay;
   print $detailstream @dev_okay;
   
   if ($mode eq "html")
     {
       print(<<END);
   </pre>
   </body>
   </html>
   END
     }
   
   # ================================================================ Subroutines.
   # Note that "vers_cmp" and "have_vers" are adapted from a bugzilla version 2.16
   # "checksetup.pl" script.
   
   # ------------ vers_cmp : compare two version numbers and see which is greater.
   # vers_cmp is adapted from Sort::Versions 1.3 1996/07/11 13:37:00 kjahds,
   # which is not included with Perl by default, hence the need to copy it here.
   # Seems silly to require it when this is the only place we need it...
   sub vers_cmp
     {
       if (@_ < 2) { die "not enough parameters for vers_cmp" }
       if (@_ > 2) { die "too many parameters for vers_cmp" }
       my ($a, $b) = @_;
       my (@A) = ($a =~ /(\.|\d+|[^\.\d]+)/g);
       my (@B) = ($b =~ /(\.|\d+|[^\.\d]+)/g);
       my ($A,$B);
       while (@A and @B)
         {
           $A = shift @A;
           $B = shift @B;
           if ($A eq "." and $B eq ".")
             {
               next;
             }
           elsif ( $A eq "." )
             {
               return -1;
             }
           elsif ( $B eq "." )
             {
               return 1;
             }
           elsif ($A =~ /^\d+$/ and $B =~ /^\d+$/)
             {
               return $A <=> $B if $A <=> $B;
             }
           else
             {
               $A = uc $A;
               $B = uc $B;
               return $A cmp $B if $A cmp $B;
             }
         }
       @A <=> @B;
     }
   
   # --------------- have_vers: syntax check the version number and call vers_cmp.
   # This was originally clipped from the libnet Makefile.PL, adapted here to
   # use the above vers_cmp routine for accurate version checking.
   sub have_vers
     {
       my ($pkg, $wanted) = @_;
       my ($msg, $vnum, $vstr);
       no strict 'refs';
       # printf("Checking for %15s %-9s ", $pkg, !$wanted?'(any)':"(v$wanted)");
   
       eval { my $p; ($p = $pkg . ".pm") =~ s!::!/!g; require $p; };
   
       $vnum = ${"${pkg}::VERSION"} || ${"${pkg}::Version"} || 0;
       $vnum = -1 if $@;
   
       if ($vnum eq "-1") # string compare just in case it's non-numeric
         {
           $vstr = "not found";
         }
       elsif (vers_cmp($vnum,"0") > -1)
         {
           $vstr = "found v$vnum";
         }
       else
         {
           $vstr = "found unknown version";
         }
   
 exit $errorflag;      my $vok = (vers_cmp($vnum,$wanted) > -1);
       # print ((($vok) ? "ok: " : " "), "$vstr\n");
       return ($vok,$vstr);
     }

Removed from v.1.1  
changed lines
  Added in v.1.10


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>