Diff for /loncom/interface/lonmeta.pm between versions 1.60 and 1.67

version 1.60, 2004/02/09 20:22:18 version 1.67, 2004/04/14 16:14:29
Line 29 Line 29
 package Apache::lonmeta;  package Apache::lonmeta;
   
 use strict;  use strict;
   use LONCAPA::lonmetadata();
 use Apache::Constants qw(:common);  use Apache::Constants qw(:common);
 use Apache::lonnet();  use Apache::lonnet();
 use Apache::loncommon();  use Apache::loncommon();
Line 39  use Apache::lonlocal; Line 40  use Apache::lonlocal;
 use Apache::lonmysql;  use Apache::lonmysql;
 use Apache::lonmsg;  use Apache::lonmsg;
   
 # MySQL table columns  
   
 my @columns;  
   
 # ----------------------------------------- Fetch and evaluate dynamic metadata  
   
   # Fetch and evaluate dynamic metadata
 sub dynamicmeta {  sub dynamicmeta {
     my $url=&Apache::lonnet::declutter(shift);      my $url=&Apache::lonnet::declutter(shift);
     $url=~s/\.meta$//;      $url=~s/\.meta$//;
Line 54  sub dynamicmeta { Line 51  sub dynamicmeta {
     $regexp='___'.$regexp.'___';      $regexp='___'.$regexp.'___';
     my %evaldata=&Apache::lonnet::dump('nohist_resevaldata',$adomain,      my %evaldata=&Apache::lonnet::dump('nohist_resevaldata',$adomain,
        $aauthor,$regexp);         $aauthor,$regexp);
     my %sum=();      my %DynamicData = &LONCAPA::lonmetadata::process_reseval_data(\%evaldata);
     my %cnt=();      my %Data = &LONCAPA::lonmetadata::process_dynamic_metadata($url,
     my %concat=();                                                                 \%DynamicData);
     my %listitems=(  
                    'course'       => 'add',  
                    'goto'         => 'add',  
                    'comefrom'     => 'add',  
                    'avetries'     => 'avg',  
                    'stdno'        => 'add',  
                    'difficulty'   => 'avg',  
                    'clear'        => 'avg',  
                    'technical'    => 'avg',  
                    'helpful'      => 'avg',  
                    'correct'      => 'avg',  
                    'depth'        => 'avg',  
                    'comments'     => 'app',  
                    'usage'        => 'cnt'  
                    );  
     while ($_=each(%evaldata)) {  
  my ($item,$purl,$cat)=split(/___/,$_);  
  ### Apache->request->print("\n".$_.' - '.$item.'<br />');  
  if (defined($cnt{$cat})) { $cnt{$cat}++; } else { $cnt{$cat}=1; }  
         unless ($listitems{$cat} eq 'app') {  
             if (defined($sum{$cat})) {  
                $sum{$cat}+=$evaldata{$_};  
                $concat{$cat}.=','.$item;  
     } else {  
                $sum{$cat}=$evaldata{$_};  
                $concat{$cat}=$item;  
     }  
         } else {  
             if (defined($sum{$cat})) {  
                if ($evaldata{$_}) {  
                   $sum{$cat}.='<hr />'.$evaldata{$_};  
        }  
      } else {  
        $sum{$cat}=''.$evaldata{$_};  
     }  
  }  
     }  
     my %returnhash=();  
     while ($_=each(%cnt)) {  
        if ($listitems{$_} eq 'avg') {  
    $returnhash{$_}=int(($sum{$_}/$cnt{$_})*100.0+0.5)/100.0;  
        } elsif ($listitems{$_} eq 'cnt') {  
            $returnhash{$_}=$cnt{$_};  
        } else {  
            $returnhash{$_}=$sum{$_};  
        }  
        $returnhash{$_.'_list'}=$concat{$_};  
        ### Apache->request->print("\n<hr />".$_.': '.$returnhash{$_}.'<br />'.$returnhash{$_.'_list'});  
     }  
     #      #
     # Deal with 'count' separately      # Deal with 'count' separately
     $returnhash{'count'} = &access_count($url,$aauthor,$adomain);      $Data{'count'} = &access_count($url,$aauthor,$adomain);
     # since "usage" is reserved word in MySQL ...      #
     $returnhash{'sequsage'}=$returnhash{'usage'};      # Debugging code I will probably need later
     $returnhash{'sequsage_list'}=$returnhash{'usage_list'};      if (0) {
           &Apache::lonnet::logthis('Dynamic Metadata');
     return %returnhash;          while(my($k,$v)=each(%Data)){
               &Apache::lonnet::logthis('    "'.$k.'"=>"'.$v.'"');
           }
           &Apache::lonnet::logthis('-------------------');
       }
       return %Data;
 }  }
   
 sub access_count {  sub access_count {
Line 127  sub access_count { Line 80  sub access_count {
     }      }
 }  }
   
 # ------------------------------------- Try to make an alt tag if there is none  # Try to make an alt tag if there is none
   
 sub alttag {  sub alttag {
     my ($base,$src)=@_;      my ($base,$src)=@_;
     my $fullpath=&Apache::lonnet::hreflocation($base,$src);      my $fullpath=&Apache::lonnet::hreflocation($base,$src);
     my $alttag=&Apache::lonnet::metadata($fullpath,'title').' '.      my $alttag=&Apache::lonnet::metadata($fullpath,'title').' '.
                &Apache::lonnet::metadata($fullpath,'subject').' '.          &Apache::lonnet::metadata($fullpath,'subject').' '.
                &Apache::lonnet::metadata($fullpath,'abstract');          &Apache::lonnet::metadata($fullpath,'abstract');
     $alttag=~s/\s+/ /gs;      $alttag=~s/\s+/ /gs;
     $alttag=~s/\"//gs;      $alttag=~s/\"//gs;
     $alttag=~s/\'//gs;      $alttag=~s/\'//gs;
     $alttag=~s/\s+$//gs;      $alttag=~s/\s+$//gs;
     $alttag=~s/^\s+//gs;      $alttag=~s/^\s+//gs;
     if ($alttag) { return $alttag; } else       if ($alttag) { 
                  { return &mt('No information available'); }          return $alttag; 
       } else { 
           return &mt('No information available'); 
       }
 }  }
   
 # -------------------------------------------------------------- Author display  # Author display
   
 sub authordisplay {  sub authordisplay {
     my ($aname,$adom)=@_;      my ($aname,$adom)=@_;
     return &Apache::loncommon::aboutmewrapper(      return &Apache::loncommon::aboutmewrapper
                 &Apache::loncommon::plainname($aname,$adom),          (&Apache::loncommon::plainname($aname,$adom),
                     $aname,$adom,'preview').' <tt>['.$aname.'@'.$adom.']</tt>';           $aname,$adom,'preview').' <tt>['.$aname.'@'.$adom.']</tt>';
 }  }
   
 # -------------------------------------------------------------- Pretty display  # Pretty display
   
 sub evalgraph {  sub evalgraph {
     my $value=shift;      my $value=shift;
     unless ($value) { return ''; }      if (! $value) { 
           return '';
       }
     my $val=int($value*10.+0.5)-10;      my $val=int($value*10.+0.5)-10;
     my $output='<table border=0 cellpadding=0 cellspacing=0><tr>';      my $output='<table border=0 cellpadding=0 cellspacing=0><tr>';
     if ($val>=20) {      if ($val>=20) {
Line 171  sub evalgraph { Line 126  sub evalgraph {
  $output.='<td width='.($val-20).' bgcolor="#33FF33">&nbsp;</td>'.   $output.='<td width='.($val-20).' bgcolor="#33FF33">&nbsp;</td>'.
                  '<td width='.(40-$val).' bgcolor="#555555">&nbsp;</td>';                   '<td width='.(40-$val).' bgcolor="#555555">&nbsp;</td>';
     } else {      } else {
        $output.='<td width=20 bgcolor="#555555">&nbsp&nbsp;</td>';          $output.='<td width=20 bgcolor="#555555">&nbsp&nbsp;</td>';
     }      }
     $output.='<td> ('.$value.') </td></tr></table>';      $output.='<td> ('.$value.') </td></tr></table>';
     return $output;      return $output;
Line 179  sub evalgraph { Line 134  sub evalgraph {
   
 sub diffgraph {  sub diffgraph {
     my $value=shift;      my $value=shift;
     unless ($value) { return ''; }      if (! $value) { 
           return '';
       }
     my $val=int(40.0*$value+0.5);      my $val=int(40.0*$value+0.5);
     my @colors=('#FF9933','#EEAA33','#DDBB33','#CCCC33',      my @colors=('#FF9933','#EEAA33','#DDBB33','#CCCC33',
                 '#BBDD33','#CCCC33','#DDBB33','#EEAA33');                  '#BBDD33','#CCCC33','#DDBB33','#EEAA33');
Line 195  sub diffgraph { Line 152  sub diffgraph {
     return $output;      return $output;
 }  }
   
 # ==================================================== Turn MySQL row into hash  
   
 sub metadata_col_to_hash {  
     my @cols=@_;  
     my %hash=();  
     for (my $i=0; $i<=$#columns; $i++) {  
  $hash{$columns[$i]}=$cols[$i];  
     }  
     return %hash;  
 }  
   
 # ============================================================= The field names  
   
   # The field names
 sub fieldnames {  sub fieldnames {
     return &Apache::lonlocal::texthash(      return &Apache::lonlocal::texthash
    'title' => 'Title',          (
    'author' =>'Author(s)',           'title' => 'Title',
    'authorspace' => 'Author Space',           'author' =>'Author(s)',
    'modifyinguser' => 'Last Modifying User',           'authorspace' => 'Author Space',
    'subject' => 'Subject',           'modifyinguser' => 'Last Modifying User',
    'keywords' => 'Keyword(s)',           'subject' => 'Subject',
    'notes' => 'Notes',           'keywords' => 'Keyword(s)',
    'abstract' => 'Abstract',           'notes' => 'Notes',
                                    'lowestgradelevel' => 'Lowest Grade Level',           'abstract' => 'Abstract',
                                    'highestgradelevel' => 'Highest Grade Level',           'lowestgradelevel' => 'Lowest Grade Level',
                                    'standards' => 'Standards',           'highestgradelevel' => 'Highest Grade Level',
    'mime' => 'MIME Type',           'standards' => 'Standards',
    'language' => 'Language',           'mime' => 'MIME Type',
    'creationdate' => 'Creation Date',           'language' => 'Language',
    'lastrevisiondate' => 'Last Revision Date',           'creationdate' => 'Creation Date',
    'owner' => 'Publisher/Owner',           'lastrevisiondate' => 'Last Revision Date',
                                    'copyright' => 'Copyright/Distribution',           'owner' => 'Publisher/Owner',
    'customdistributionfile' => 'Custom Distribution File',           'copyright' => 'Copyright/Distribution',
                                    'obsolete' => 'Obsolete',           'customdistributionfile' => 'Custom Distribution File',
    'obsoletereplacement' => 'Suggested Replacement for Obsolete File',           'obsolete' => 'Obsolete',
    'count'      => 'Network-wide number of accesses (hits)',           'obsoletereplacement' => 'Suggested Replacement for Obsolete File',
    'course'     => 'Network-wide number of courses using resource',           'count'      => 'Network-wide number of accesses (hits)',
    'course_list' => 'Network-wide courses using resource',           'course'     => 'Network-wide number of courses using resource',
    'sequsage'      => 'Number of resources using or importing resource',           'course_list' => 'Network-wide courses using resource',
    'sequsage_list' => 'Resources using or importing resource',           'sequsage'      => 'Number of resources using or importing resource',
    'goto'       => 'Number of resources that follow this resource in maps',           'sequsage_list' => 'Resources using or importing resource',
    'goto_list'  => 'Resources that follow this resource in maps',           'goto'       => 'Number of resources that follow this resource in maps',
    'comefrom'   => 'Number of resources that lead up to this resource in maps',           'goto_list'  => 'Resources that follow this resource in maps',
    'comefrom_list' => 'Resources that lead up to this resource in maps',           'comefrom'   => 'Number of resources that lead up to this resource in maps',
    'clear'      => 'Material presented in clear way',           'comefrom_list' => 'Resources that lead up to this resource in maps',
    'depth'      => 'Material covered with sufficient depth',           'clear'      => 'Material presented in clear way',
    'helpful'    => 'Material is helpful',           'depth'      => 'Material covered with sufficient depth',
    'correct'    => 'Material appears to be correct',           'helpful'    => 'Material is helpful',
    'technical'  => 'Resource is technically correct',            'correct'    => 'Material appears to be correct',
    'avetries'   => 'Average number of tries till solved',           'technical'  => 'Resource is technically correct', 
    'stdno'      => 'Total number of students who have worked on this problem',           'avetries'   => 'Average number of tries till solved',
    'difficulty' => 'Degree of difficulty'           'stdno'      => 'Total number of students who have worked on this problem',
        );           'difficulty' => 'Degree of difficulty'
            );
 }  }
   
 # =========================================== Pretty printing of metadata field  # Pretty printing of metadata field
   
 sub prettyprint {  sub prettyprint {
     my ($type,$value)=@_;      my ($type,$value)=@_;
     unless (defined($value)) { return '&nbsp;'; }      if (! defined($value)) { 
 # Title          return '&nbsp;'; 
       }
       # Title
     if ($type eq 'title') {      if ($type eq 'title') {
  return '<font size="+1" face="arial">'.$value.'</font>';   return '<font size="+1" face="arial">'.$value.'</font>';
     }      }
 # Dates      # Dates
     if (($type eq 'creationdate') ||      if (($type eq 'creationdate') ||
  ($type eq 'lastrevisiondate')) {   ($type eq 'lastrevisiondate')) {
  return ($value?&Apache::lonlocal::locallocaltime(   return ($value?&Apache::lonlocal::locallocaltime(
   &Apache::lonmysql::unsqltime($value)):    &Apache::lonmysql::unsqltime($value)):
  &mt('not available'));   &mt('not available'));
     }      }
 # Language      # Language
     if ($type eq 'language') {      if ($type eq 'language') {
  return &Apache::loncommon::languagedescription($value);   return &Apache::loncommon::languagedescription($value);
     }      }
 # Copyright      # Copyright
     if ($type eq 'copyright') {      if ($type eq 'copyright') {
  return &Apache::loncommon::copyrightdescription($value);   return &Apache::loncommon::copyrightdescription($value);
     }      }
 # MIME      # MIME
     if ($type eq 'mime') {      if ($type eq 'mime') {
        return '<img src="'.&Apache::loncommon::icon($value).'" />&nbsp;'.          return '<img src="'.&Apache::loncommon::icon($value).'" />&nbsp;'.
    &Apache::loncommon::filedescription($value);              &Apache::loncommon::filedescription($value);
    }      }
 # Person      # Person
     if (($type eq 'author') ||       if (($type eq 'author') || 
  ($type eq 'owner') ||   ($type eq 'owner') ||
  ($type eq 'modifyinguser') ||   ($type eq 'modifyinguser') ||
Line 287  sub prettyprint { Line 236  sub prettyprint {
  $value=~s/(\w+)(\:|\@)(\w+)/&authordisplay($1,$3)/gse;   $value=~s/(\w+)(\:|\@)(\w+)/&authordisplay($1,$3)/gse;
  return $value;   return $value;
     }      }
 # Gradelevel      # Gradelevel
     if (($type eq 'lowestgradelevel') ||      if (($type eq 'lowestgradelevel') ||
  ($type eq 'highestgradelevel')) {   ($type eq 'highestgradelevel')) {
  return &Apache::loncommon::gradeleveldescription($value);   return &Apache::loncommon::gradeleveldescription($value);
     }      }
 # Only for advance users below      # Only for advance users below
     unless ($ENV{'user.adv'}) { return '<i>- '.&mt('not displayed').' -</i>' };      if (! $ENV{'user.adv'}) { 
 # File          return '<i>- '.&mt('not displayed').' -</i>';
       }
       # File
     if (($type eq 'customdistributionfile') ||      if (($type eq 'customdistributionfile') ||
  ($type eq 'obsoletereplacement') ||   ($type eq 'obsoletereplacement') ||
  ($type eq 'goto_list') ||   ($type eq 'goto_list') ||
  ($type eq 'comefrom_list') ||   ($type eq 'comefrom_list') ||
  ($type eq 'sequsage_list')) {   ($type eq 'sequsage_list')) {
  return join('<br />',map {   return join('<br />',map {
        my $url=&Apache::lonnet::clutter($_);              my $url=&Apache::lonnet::clutter($_);
        '<br /><b>'.&Apache::lonnet::gettitle($url).'</b>'.              '<b>'.&Apache::lonnet::gettitle($url).'</b>'.
        &Apache::lonhtmlcommon::crumbs($url,'preview','');                  &Apache::lonhtmlcommon::crumbs($url,'preview','',undef,'+0');
     } split(/\s*\,\s*/,$value));          } split(/\s*\,\s*/,$value));
     }      }
 # Evaluations      # Evaluations
     if (($type eq 'clear') ||      if (($type eq 'clear') ||
  ($type eq 'depth') ||   ($type eq 'depth') ||
  ($type eq 'helpful') ||   ($type eq 'helpful') ||
Line 314  sub prettyprint { Line 265  sub prettyprint {
  ($type eq 'technical')) {   ($type eq 'technical')) {
  return &evalgraph($value);   return &evalgraph($value);
     }      }
 # Difficulty      # Difficulty
     if ($type eq 'difficulty') {      if ($type eq 'difficulty') {
  return &diffgraph($value);   return &diffgraph($value);
     }      }
 # List of courses      # List of courses
     if ($type=~/\_list/) {      if ($type=~/\_list/) {
  return join('<br />',map {   return join('<br />',map {
     my %courseinfo=&Apache::lonnet::coursedescription($_);        my %courseinfo=&Apache::lonnet::coursedescription($_);  
Line 327  sub prettyprint { Line 278  sub prettyprint {
  $courseinfo{'description'}.'</a>';   $courseinfo{'description'}.'</a>';
  } split(/\s*\,\s*/,$value));   } split(/\s*\,\s*/,$value));
     }      }
 # No pretty print found      # No pretty print found
     return $value;      return $value;
 }  }
 # ============================================== Pretty input of metadata field  
   
   # Pretty input of metadata field
 sub direct {  sub direct {
     return shift;      return shift;
 }  }
   
 sub selectbox {  sub selectbox {
     my ($name,$value,$functionref,@idlist)=@_;      my ($name,$value,$functionref,@idlist)=@_;
     unless (defined($functionref)) { $functionref=\&direct; }      if (! defined($functionref)) {
           $functionref=\&direct;
       }
     my $selout='<select name="'.$name.'">';      my $selout='<select name="'.$name.'">';
     foreach (@idlist) {      foreach (@idlist) {
         $selout.='<option value=\''.$_.'\'';          $selout.='<option value=\''.$_.'\'';
Line 352  sub selectbox { Line 305  sub selectbox {
   
 sub relatedfield {  sub relatedfield {
     my ($show,$relatedsearchflag,$relatedsep,$fieldname,$relatedvalue)=@_;      my ($show,$relatedsearchflag,$relatedsep,$fieldname,$relatedvalue)=@_;
     unless ($relatedsearchflag) { return ''; }      if (! $relatedsearchflag) { 
     unless (defined($relatedsep)) { $relatedsep=' '; }          return '';
     unless ($show) { return $relatedsep.'&nbsp;'; }      }
       if (! defined($relatedsep)) {
           $relatedsep=' ';
       }
       if (! $show) {
           return $relatedsep.'&nbsp;';
       }
     return $relatedsep.'<input type="checkbox" name="'.$fieldname.'_related"'.      return $relatedsep.'<input type="checkbox" name="'.$fieldname.'_related"'.
  ($relatedvalue?' checked="1"':'').' />';   ($relatedvalue?' checked="1"':'').' />';
 }  }
Line 362  sub relatedfield { Line 321  sub relatedfield {
 sub prettyinput {  sub prettyinput {
     my ($type,$value,$fieldname,$formname,      my ($type,$value,$fieldname,$formname,
  $relatedsearchflag,$relatedsep,$relatedvalue)=@_;   $relatedsearchflag,$relatedsep,$relatedvalue)=@_;
 # Language      # Language
     if ($type eq 'language') {      if ($type eq 'language') {
  return &selectbox($fieldname,   return &selectbox($fieldname,
   $value,    $value,
   \&Apache::loncommon::languagedescription,    \&Apache::loncommon::languagedescription,
   (&Apache::loncommon::languageids)).    (&Apache::loncommon::languageids)).
        &relatedfield(0,$relatedsearchflag,$relatedsep);                                &relatedfield(0,$relatedsearchflag,$relatedsep);
     }      }
 # Copyright      # Copyright
     if ($type eq 'copyright') {      if ($type eq 'copyright') {
  return &selectbox($fieldname,   return &selectbox($fieldname,
   $value,    $value,
   \&Apache::loncommon::copyrightdescription,    \&Apache::loncommon::copyrightdescription,
   (&Apache::loncommon::copyrightids)).    (&Apache::loncommon::copyrightids)).
        &relatedfield(0,$relatedsearchflag,$relatedsep);                                &relatedfield(0,$relatedsearchflag,$relatedsep);
     }      }
 # Gradelevels      # Gradelevels
     if (($type eq 'lowestgradelevel') ||      if (($type eq 'lowestgradelevel') ||
  ($type eq 'highestgradelevel')) {   ($type eq 'highestgradelevel')) {
  return &Apache::loncommon::select_level_form($value,$fieldname).   return &Apache::loncommon::select_level_form($value,$fieldname).
        &relatedfield(0,$relatedsearchflag,$relatedsep);              &relatedfield(0,$relatedsearchflag,$relatedsep);
     }      }
 # Obsolete      # Obsolete
     if ($type eq 'obsolete') {      if ($type eq 'obsolete') {
  return '<input type="checkbox" name="'.$fieldname.'"'.   return '<input type="checkbox" name="'.$fieldname.'"'.
     ($value?' checked="1"':'').' />'.      ($value?' checked="1"':'').' />'.
        &relatedfield(0,$relatedsearchflag,$relatedsep);               &relatedfield(0,$relatedsearchflag,$relatedsep); 
     }      }
 # Obsolete replacement file      # Obsolete replacement file
     if ($type eq 'obsoletereplacement') {      if ($type eq 'obsoletereplacement') {
  return '<input type="text" name="'.$fieldname.   return '<input type="text" name="'.$fieldname.
     '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.      '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.
     "('".$formname."','".$fieldname."'".      "('".$formname."','".$fieldname."'".
     ",'')\">".&mt('Select').'</a>'.      ",'')\">".&mt('Select').'</a>'.
        &relatedfield(0,$relatedsearchflag,$relatedsep);               &relatedfield(0,$relatedsearchflag,$relatedsep); 
    }      }
 # Customdistribution file      # Customdistribution file
     if ($type eq 'customdistributionfile') {      if ($type eq 'customdistributionfile') {
  return '<input type="text" name="'.$fieldname.   return '<input type="text" name="'.$fieldname.
     '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.      '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.
     "('".$formname."','".$fieldname."'".      "('".$formname."','".$fieldname."'".
     ",'rights')\">".&mt('Select').'</a>'.      ",'rights')\">".&mt('Select').'</a>'.
        &relatedfield(0,$relatedsearchflag,$relatedsep);               &relatedfield(0,$relatedsearchflag,$relatedsep); 
     }      }
 # Dates      # Dates
     if (($type eq 'creationdate') ||      if (($type eq 'creationdate') ||
  ($type eq 'lastrevisiondate')) {   ($type eq 'lastrevisiondate')) {
  return &Apache::lonhtmlcommon::date_setter($formname,   return 
    $fieldname,$value).              &Apache::lonhtmlcommon::date_setter($formname,$fieldname,$value).
        &relatedfield(0,$relatedsearchflag,$relatedsep);              &relatedfield(0,$relatedsearchflag,$relatedsep);
     }      }
 # No pretty input found      # No pretty input found
     $value=~s/^\s+//gs;      $value=~s/^\s+//gs;
     $value=~s/\s+$//gs;      $value=~s/\s+$//gs;
     $value=~s/\s+/ /gs;      $value=~s/\s+/ /gs;
     $value=~s/\"/\&quod\;/gs;      $value=~s/\"/\&quod\;/gs;
     return       return 
     '<input type="text" name="'.$fieldname.'" size="80" value="'.$value.'" />'.          '<input type="text" name="'.$fieldname.'" size="80" '.
     &relatedfield(1,$relatedsearchflag,$relatedsep,$fieldname,$relatedvalue);           'value="'.$value.'" />'.
           &relatedfield(1,$relatedsearchflag,$relatedsep,$fieldname,
                         $relatedvalue); 
 }  }
   
 # ================================================================ Main Handler  # Main Handler
   
 sub handler {  sub handler {
   my $r=shift;      my $r=shift;
       #
     my $loaderror=&Apache::lonnet::overloaderror($r);  
     if ($loaderror) { return $loaderror; }  
   
   
     my $uri=$r->uri;      my $uri=$r->uri;
       #
 # ====================================================== Looking for all bombs?      # Check to see if this server is overloaded
   if ($uri=~/\/adm\/bombs\/(.*)$/) {      my $loaderror=&Apache::lonnet::overloaderror($r);
 # ----------------------------------------------------------- Set document type      if ($loaderror) { 
       $uri=&Apache::lonnet::declutter($1);          return $loaderror;
       &Apache::loncommon::content_type($r,'text/html');      }
       $r->send_http_header;      #
             # Check to see if original resource server is overloaded
       return OK if $r->header_only;  
       $r->print(&Apache::loncommon::bodytag('Error Messages'));  
       $r->print('<h1>'.&Apache::lonnet::clutter($uri).'</h1>');  
       my ($domain,$author)=($uri=~/^(\w+)\/(\w+)\//);  
       if (&Apache::loncacc::constructaccess('/~'.$author.'/',$domain)) {  
   my %brokenurls=&Apache::lonmsg::all_url_author_res_msg($author,$domain);  
   foreach (sort keys %brokenurls) {  
       if ($_=~/^\Q$uri\E/) {  
   $r->print(&Apache::lonhtmlcommon::crumbs(&Apache::lonnet::clutter($_)).  
     &Apache::lonmsg::retrieve_author_res_msg($_).'<hr />');  
       }  
   }  
       } else {  
   $r->print(&mt('Not authorized'));  
       }  
       $r->print('</body></html>');  
   } elsif ($uri!~/^\/\~/) {   
 # =========================================== This is not in construction space  
     my ($resdomain,$resuser)=      my ($resdomain,$resuser)=
            (&Apache::lonnet::declutter($uri)=~/^(\w+)\/(\w+)\//);          (&Apache::lonnet::declutter($uri)=~/^(\w+)\/(\w+)\//);
       $loaderror=&Apache::lonnet::overloaderror
     $loaderror=          ($r,&Apache::lonnet::homeserver($resuser,$resdomain));
        &Apache::lonnet::overloaderror($r,      if ($loaderror) { 
          &Apache::lonnet::homeserver($resuser,$resdomain));          return $loaderror;
     if ($loaderror) { return $loaderror; }      }
       #
   my %content=();      # Set document type
       &Apache::loncommon::content_type($r,'text/html');
 # ----------------------------------------------------------- Set document type      $r->send_http_header;
       return OK if $r->header_only;
   &Apache::loncommon::content_type($r,'text/html');      #
   $r->send_http_header;      $r->print('<html><head><title>'.
                 'Catalog Information'.
   return OK if $r->header_only;                '</title></head>');
       if ($uri=~m:/adm/bombs/(.*)$:) {
           $r->print(&Apache::loncommon::bodytag('Error Messages'));
           # Looking for all bombs?
           &report_bombs($r,$uri);
       } elsif ($uri=~/^\/\~/) { 
           # Construction space
           $r->print(&Apache::loncommon::bodytag
                     ('Edit Catalog Information','','','',$resdomain));
           &present_editable_metadata($r,$uri);
       } else {
           $r->print(&Apache::loncommon::bodytag
                     ('Catalog Information','','','',$resdomain));
           &present_uneditable_metadata($r,$uri);
       }
       $r->print('</body></html>');
       return OK;
   }
   
   #####################################################
   #####################################################
   ###                                               ###
   ###                Report Bombs                   ###
   ###                                               ###
   #####################################################
   #####################################################
   sub report_bombs {
       my ($r,$uri) = @_;
       # Set document type
       $uri =~ s:/adm/bombs/::;
       $uri = &Apache::lonnet::declutter($uri);
       $r->print('<h1>'.&Apache::lonnet::clutter($uri).'</h1>');
       my ($domain,$author)=($uri=~/^(\w+)\/(\w+)\//);
       if (&Apache::loncacc::constructaccess('/~'.$author.'/',$domain)) {
           my %brokenurls = 
               &Apache::lonmsg::all_url_author_res_msg($author,$domain);
           foreach (sort(keys(%brokenurls))) {
               if ($_=~/^\Q$uri\E/) {
                   $r->print(&Apache::lonhtmlcommon::crumbs
                             (&Apache::lonnet::clutter($_)).
                             &Apache::lonmsg::retrieve_author_res_msg($_).
                             '<hr />');
               }
           }
       } else {
           $r->print(&mt('Not authorized'));
       }
       return;
   }
   
 # ------------------------------------------------------------------- Read file  #####################################################
   foreach (split(/\,/,&Apache::lonnet::metadata($uri,'keys'))) {  #####################################################
       $content{$_}=&Apache::lonnet::metadata($uri,$_);  ###                                               ###
   }  ###        Uneditable Metadata Display            ###
 # --------------------------------------------------------------- Render Output  ###                                               ###
 # displayed url  #####################################################
   #####################################################
   sub present_uneditable_metadata {
       my ($r,$uri) = @_;
       #
       my %content=();
       # Read file
       foreach (split(/\,/,&Apache::lonnet::metadata($uri,'keys'))) {
           $content{$_}=&Apache::lonnet::metadata($uri,$_);
       }
       # Render Output
       # displayed url
     my ($thisversion)=($uri=~/\.(\d+)\.(\w+)\.meta$/);      my ($thisversion)=($uri=~/\.(\d+)\.(\w+)\.meta$/);
     $uri=~s/\.meta$//;      $uri=~s/\.meta$//;
     my $disuri=&Apache::lonnet::clutter($uri);      my $disuri=&Apache::lonnet::clutter($uri);
 # version      # version
     my $currentversion=&Apache::lonnet::getversion($disuri);      my $currentversion=&Apache::lonnet::getversion($disuri);
     my $versiondisplay='';      my $versiondisplay='';
     if ($thisversion) {      if ($thisversion) {
  $versiondisplay=&mt('Version').': '.$thisversion.          $versiondisplay=&mt('Version').': '.$thisversion.
     ' ('.&mt('most recent version').': '.              ' ('.&mt('most recent version').': '.
     ($currentversion>0?$currentversion:&mt('information not available')).')';              ($currentversion>0 ? 
                $currentversion   :
                &mt('information not available')).')';
     } else {      } else {
  $versiondisplay='Version: '.$currentversion;          $versiondisplay='Version: '.$currentversion;
     }      }
 # crumbify displayed URL      # crumbify displayed URL
     $disuri=&Apache::lonhtmlcommon::crumbs($disuri);      $disuri=&Apache::lonhtmlcommon::crumbs($disuri);
 # obsolete      # obsolete
     my $obsolete=$content{'obsolete'};      my $obsolete=$content{'obsolete'};
     my $obsoletewarning='';      my $obsoletewarning='';
     if (($obsolete) && ($ENV{'user.adv'})) {      if (($obsolete) && ($ENV{'user.adv'})) {
  $obsoletewarning='<p><font color="red">'.          $obsoletewarning='<p><font color="red">'.
     &mt('This resource has been marked obsolete by the author(s)').'</font></p>';              &mt('This resource has been marked obsolete by the author(s)').
               '</font></p>';
     }      }
       #
     my %lt=&fieldnames();      my %lt=&fieldnames();
     my $table='';      my $table='';
     my $bodytag=&Apache::loncommon::bodytag  
             ('Catalog Information','','','',$resdomain);  
     foreach ('title',       foreach ('title', 
      'author',                'author', 
      'subject',                'subject', 
      'keywords',                'keywords', 
      'notes',                'notes', 
      'abstract',               'abstract',
      'lowestgradelevel',               'lowestgradelevel',
      'highestgradelevel',               'highestgradelevel',
      'standards',                'standards', 
      'mime',                'mime', 
      'language',                'language', 
      'creationdate',                'creationdate', 
      'lastrevisiondate',                'lastrevisiondate', 
      'owner',                'owner', 
      'copyright',                'copyright', 
      'customdistributionfile',                'customdistributionfile', 
      'obsolete',                'obsolete', 
      'obsoletereplacement') {               'obsoletereplacement') {
  $table.='<tr><td bgcolor="#AAAAAA">'.$lt{$_}.          $table.='<tr><td bgcolor="#AAAAAA">'.$lt{$_}.
         '</td><td bgcolor="#CCCCCC">'.              '</td><td bgcolor="#CCCCCC">'.
                 &prettyprint($_,$content{$_}).'</td></tr>';              &prettyprint($_,$content{$_}).'</td></tr>';
  delete $content{$_};          delete $content{$_};
     }      }
       #
   $r->print(<<ENDHEAD);      $r->print(<<ENDHEAD);
 <html><head><title>Catalog Information</title></head>  
 $bodytag  
 <h2>$content{'title'}</h2>  <h2>$content{'title'}</h2>
 <h3><tt>$disuri</tt></h3>  <h3><tt>$disuri</tt></h3>
 $obsoletewarning  $obsoletewarning
Line 544  $versiondisplay<br /> Line 540  $versiondisplay<br />
 $table  $table
 </table>  </table>
 ENDHEAD  ENDHEAD
   if ($ENV{'user.adv'}) {      if ($ENV{'user.adv'}) {
 # ------------------------------------------------------------ Dynamic Metadata          &print_dynamic_metadata($r,$uri);
       $r->print(      }
  '<h3>'.&mt('Dynamic Metadata').' ('.      return;
  &mt('updated periodically').')</h3>'.&mt('Processing').  }
  ' ...<br />');  
       $r->rflush();  sub print_dynamic_metadata {
       my %items=&fieldnames();      my ($r,$uri) = @_;
       my %dynmeta=&dynamicmeta($uri);      #
 # General Access and Usage Statistics      my $description = 'Dynamic Metadata (updated periodically)';
       $r->print('<h4>'.&mt('Access and Usage Statistics').'</h4><table cellspacing=2 border=0>');      $r->print('<h3>'.&mt($description).'</h3>'.
       foreach ('count',                &mt('Processing').' ...<br />');
        'sequsage','sequsage_list',      $r->rflush();
        'comefrom','comefrom_list',      my %items=&fieldnames();
        'goto','goto_list',      my %dynmeta=&dynamicmeta($uri);
        'course','course_list') {      #
   $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td><td bgcolor="#CCCCCC">'.      # General Access and Usage Statistics
     &prettyprint($_,$dynmeta{$_})."</td></tr>\n");      $r->print('<h4>'.&mt('Access and Usage Statistics').'</h4>'.
       }                '<table cellspacing=2 border=0>');
       $r->print('</table>');      foreach ('count',
       if ($uri=~/\.(problem|exam|quiz|assess|survey|form)\.meta$/) {               'sequsage','sequsage_list',
 # This is an assessment, print assessment data               'comefrom','comefrom_list',
   $r->print(               'goto','goto_list',
     '<h4>'.&mt('Assessment Statistical Data').'</h4><table cellspacing=2 border=0>');               'course','course_list') {
   foreach ('stdno','avetries','difficulty') {          $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
       $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td><td bgcolor="#CCCCCC">'.                    '<td bgcolor="#CCCCCC">'.
  &prettyprint($_,$dynmeta{$_})."</td></tr>\n");                    &prettyprint($_,$dynmeta{$_})."</td></tr>\n");
   }      }
   $r->print('</table>');          $r->print('</table>');
       }      if ($uri=~/\.(problem|exam|quiz|assess|survey|form)$/) {
       $r->print('<h4>'.&mt('Evaluation Data').'</h4><table cellspacing=2 border=0>');          # This is an assessment, print assessment data
       foreach ('clear','depth','helpful','correct','technical') {          $r->print(
   $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td><td bgcolor="#CCCCCC">'.                    '<h4>'.&mt('Assessment Statistical Data').'</h4>'.
     &prettyprint($_,$dynmeta{$_})."</td></tr>\n");                    '<table cellspacing=2 border=0>');
      }              foreach ('stdno','avetries','difficulty') {
       $r->print('</table>');              $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
       $uri=~/^\/res\/(\w+)\/(\w+)\//;                         '<td bgcolor="#CCCCCC">'.
       if ((($ENV{'user.domain'} eq $1) && ($ENV{'user.name'} eq $2))                        &prettyprint($_,$dynmeta{$_})."</td></tr>\n");
   || ($ENV{'user.role.ca./'.$1.'/'.$2})) {          }
   $r->print('<h4>'.&mt('Evaluation Comments').' ('.          $r->print('</table>');    
     &mt('visible to author and co-authors only').')</h4>'.      }
     '<blockquote>'.$dynmeta{'comments'}.'</blockquote>');      
   $r->print('<a name="bombs" /><h4>'.&mt('Error Messages').' ('.      $r->print('<h4>'.&mt('Evaluation Data').'</h4>'.
     &mt('visible to author and co-authors only').')</h4>'.                '<table cellspacing=2 border=0>');
     &Apache::lonmsg::retrieve_author_res_msg($uri));      foreach ('clear','depth','helpful','correct','technical') {
       }          $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
 # ------------------------------------------------------------- All other stuff                    '<td bgcolor="#CCCCCC">'.
       $r->print(                    &prettyprint($_,$dynmeta{$_})."</td></tr>\n");
  '<h3>'.&mt('Additional Metadata (non-standard, parameters, exports)').'</h3>');      }
       foreach (sort keys %content) {      $r->print('</table>');
   my $name=$_;      $uri=~/^\/res\/(\w+)\/(\w+)\//; 
   unless ($name=~/\.display$/) {      if ((($ENV{'user.domain'} eq $1) && ($ENV{'user.name'} eq $2))
       my $display=&Apache::lonnet::metadata($uri,$name.'.display');          || ($ENV{'user.role.ca./'.$1.'/'.$2})) {
       unless ($display) { $display=$name; };          $r->print('<h4>'.&mt('Evaluation Comments').' ('.
       my $otherinfo='';                    &mt('visible to author and co-authors only').
       foreach ('name','part','type','default') {                    ')</h4>'.
   if (defined(&Apache::lonnet::metadata($uri,$name.'.'.$_))) {                    '<blockquote>'.$dynmeta{'comments'}.'</blockquote>');
       $otherinfo.=' '.$_.'='.          $r->print('<a name="bombs" /><h4>'.&mt('Error Messages').' ('.
   &Apache::lonnet::metadata($uri,$name.'.'.$_).'; ';                    &mt('visible to author and co-authors only').')'.
   }                    '</h4>'.
       }                    &Apache::lonmsg::retrieve_author_res_msg($uri));
       $r->print('<b>'.$display.':</b> '.$content{$name});      }
       if ($otherinfo) {      # All other stuff
   $r->print(' ('.$otherinfo.')');      $r->print('<h3>'.
       }                &mt('Additional Metadata (non-standard, parameters, exports)').
       $r->print("<br />\n");                '</h3>');
   }      foreach (sort(keys(%content))) {
       }          my $name=$_;
   }          if ($name!~/\.display$/) {
 # ===================================================== End Resource Space Call              my $display=&Apache::lonnet::metadata($uri,
  } else {                                                    $name.'.display');
 # ===================================================== Construction Space Call              if (! $display) { 
                   $display=$name;
 # ----------------------------------------------------------- Set document type              };
               my $otherinfo='';
   &Apache::loncommon::content_type($r,'text/html');              foreach ('name','part','type','default') {
   $r->send_http_header;                  if (defined(&Apache::lonnet::metadata($uri,
                                                         $name.'.'.$_))) {
   return OK if $r->header_only;                      $otherinfo.=' '.$_.'='.
 # ---------------------------------------------------------------------- Header                          &Apache::lonnet::metadata($uri,
   my $disuri=$uri;                                                    $name.'.'.$_).'; ';
   my $fn=&Apache::lonnet::filelocation('',$uri);                  }
   $disuri=~s/^\/\~/\/priv\//;              }
   $disuri=~s/\.meta$//;              $r->print('<b>'.$display.':</b> '.$content{$name});
   my $target=$uri;              if ($otherinfo) {
   $target=~s/^\/\~/\/res\/$ENV{'request.role.domain'}\//;                  $r->print(' ('.$otherinfo.')');
   $target=~s/\.meta$//;              }
   my $bombs=&Apache::lonmsg::retrieve_author_res_msg($target);              $r->print("<br />\n");
   if ($bombs) {          }
       if ($ENV{'form.delmsg'}) {      }
   if (&Apache::lonmsg::del_url_author_res_msg($target) eq 'ok') {      return;
       $bombs=&mt('Messages deleted.');  }
   } else {  
       $bombs=&mt('Error deleting messages');  #####################################################
   }  #####################################################
       }  ###                                               ###
       my $bodytag=&Apache::loncommon::bodytag('Error Messages');  ###          Editable metadata display            ###
       my $del=&mt('Delete Messages');  ###                                               ###
       $r->print(<<ENDBOMBS);  #####################################################
 <html><head><title>Edit Catalog Information</title></head>  #####################################################
 $bodytag  sub present_editable_metadata {
       my ($r,$uri) = @_;
       # Construction Space Call
       # Header
       my $disuri=$uri;
       my $fn=&Apache::lonnet::filelocation('',$uri);
       $disuri=~s/^\/\~/\/priv\//;
       $disuri=~s/\.meta$//;
       my $target=$uri;
       $target=~s/^\/\~/\/res\/$ENV{'request.role.domain'}\//;
       $target=~s/\.meta$//;
       my $bombs=&Apache::lonmsg::retrieve_author_res_msg($target);
       if ($bombs) {
           if ($ENV{'form.delmsg'}) {
               if (&Apache::lonmsg::del_url_author_res_msg($target) eq 'ok') {
                   $bombs=&mt('Messages deleted.');
               } else {
                   $bombs=&mt('Error deleting messages');
               }
           }
           my $del=&mt('Delete Messages');
           $r->print(<<ENDBOMBS);
 <h1>$disuri</h1>  <h1>$disuri</h1>
 <form method="post" name="defaultmeta">  <form method="post" name="defaultmeta">
 <input type="submit" name="delmsg" value="$del" />  <input type="submit" name="delmsg" value="$del" />
 <br />$bombs  <br />$bombs
 </form>  
 </body>  
 </html>  
 ENDBOMBS  ENDBOMBS
   } else {      } else {
       my $displayfile='Catalog Information for '.$disuri;          my $displayfile='Catalog Information for '.$disuri;
       if ($disuri=~/\/default$/) {          if ($disuri=~/\/default$/) {
   my $dir=$disuri;              my $dir=$disuri;
   $dir=~s/default$//;              $dir=~s/default$//;
   $displayfile=&mt('Default Cataloging Information for Directory').' '.              $displayfile=
       $dir;                  &mt('Default Cataloging Information for Directory').' '.
       }                  $dir;
       my $bodytag=&Apache::loncommon::bodytag('Edit Catalog Information');          }
       %Apache::lonpublisher::metadatafields=();          my $bodytag=
       %Apache::lonpublisher::metadatakeys=();              &Apache::loncommon::bodytag('Edit Catalog Information');
       &Apache::lonpublisher::metaeval(&Apache::lonnet::getfile($fn));          %Apache::lonpublisher::metadatafields=();
       $r->print(<<ENDEDIT);          %Apache::lonpublisher::metadatakeys=();
           &Apache::lonpublisher::metaeval(&Apache::lonnet::getfile($fn));
           $r->print(<<ENDEDIT);
 <html><head><title>Edit Catalog Information</title></head>  <html><head><title>Edit Catalog Information</title></head>
 $bodytag  $bodytag
 <h1>$displayfile</h1>  <h1>$displayfile</h1>
 <form method="post" name="defaultmeta">  <form method="post" name="defaultmeta">
 ENDEDIT  ENDEDIT
       $r->print('<script language="JavaScript">'.          $r->print('<script language="JavaScript">'.
  &Apache::loncommon::browser_and_searcher_javascript.                    &Apache::loncommon::browser_and_searcher_javascript.
  '</script>');                    '</script>');
       my %lt=&fieldnames();          my %lt=&fieldnames();
       foreach ('author','title','subject','keywords','abstract','notes',          foreach ('author','title','subject','keywords','abstract','notes',
        'copyright','customdistributionfile','language','standards',                   'copyright','customdistributionfile','language',
        'lowestgradelevel','highestgradelevel',                   'standards',
        'obsolete','obsoletereplacement') {                   'lowestgradelevel','highestgradelevel',
   if (defined($ENV{'form.new_'.$_})) {                   'obsolete','obsoletereplacement') {
       $Apache::lonpublisher::metadatafields{$_}=$ENV{'form.new_'.$_};              if (defined($ENV{'form.new_'.$_})) {
   }                  $Apache::lonpublisher::metadatafields{$_}=
   unless ($Apache::lonpublisher::metadatafields{'copyright'}) {                      $ENV{'form.new_'.$_};
       $Apache::lonpublisher::metadatafields{'copyright'}='default';              }
   }              if (! $Apache::lonpublisher::metadatafields{'copyright'}) {
   $r->print('<p>'.$lt{$_}.': '.&prettyinput($_,                  $Apache::lonpublisher::metadatafields{'copyright'}=
     $Apache::lonpublisher::metadatafields{$_},                      'default';
     'new_'.$_,'defaultmeta').'</p>');              }
       }              $r->print('<p>'.$lt{$_}.': '.
       if ($ENV{'form.store'}) {                        &prettyinput
   my $mfh;                        ($_,$Apache::lonpublisher::metadatafields{$_},
   unless ($mfh=Apache::File->new('>'.$fn)) {                         'new_'.$_,'defaultmeta').'</p>');
       $r->print(          }
  '<p><font color=red>'.&mt('Could not write metadata').', '.          if ($ENV{'form.store'}) {
  &mt('FAIL').'</font>');              my $mfh;
   } else {              if (!  ($mfh=Apache::File->new('>'.$fn))) {
       foreach (sort keys %Apache::lonpublisher::metadatafields) {                  $r->print('<p><font color=red>'.
   unless ($_=~/\./) {                            &mt('Could not write metadata').', '.
       my $unikey=$_;                            &mt('FAIL').'</font>');
       $unikey=~/^([A-Za-z]+)/;              } else {
       my $tag=$1;                  foreach (sort keys %Apache::lonpublisher::metadatafields) {
       $tag=~tr/A-Z/a-z/;                      next if ($_ =~ /\./);
       print $mfh "\n\<$tag";                      my $unikey=$_;
       foreach                       $unikey=~/^([A-Za-z]+)/;
   (split(/\,/,$Apache::lonpublisher::metadatakeys{$unikey})) {                      my $tag=$1;
       my $value=                      $tag=~tr/A-Z/a-z/;
   $Apache::lonpublisher::metadatafields{$unikey.'.'.$_};                      print $mfh "\n\<$tag";
       $value=~s/\"/\'\'/g;                      foreach (split(/\,/,
       print $mfh ' '.$_.'="'.$value.'"';                                   $Apache::lonpublisher::metadatakeys{$unikey})
   }                               ) {
       print $mfh '>'.                          my $value=
   &HTML::Entities::encode($Apache::lonpublisher::metadatafields{$unikey})                           $Apache::lonpublisher::metadatafields{$unikey.'.'.$_};
   .'</'.$tag.'>';                          $value=~s/\"/\'\'/g;
   }                          print $mfh ' '.$_.'="'.$value.'"';
       }                      }
       $r->print('<p>'.&mt('Wrote Metadata'));                      print $mfh '>'.
   }                          &HTML::Entities::encode
       }                          ($Apache::lonpublisher::metadatafields{$unikey},
       $r->print(                           '<>&"').
  '<br /><input type="submit" name="store" value="'.                           '</'.$tag.'>';
  &mt('Store Catalog Information').'"></form></body></html>');                  }
    }                  $r->print('<p>'.&mt('Wrote Metadata'));
   }              }
   return OK;          }
           $r->print('<br /><input type="submit" name="store" value="'.
                     &mt('Store Catalog Information').'">');
       }
       $r->print('</form>');
       return;
 }  }
   
 # ================================================================= BEGIN Block  ##############################################################
   ##############################################################
   # MySQL table columns
   
   my @columns;
   
 BEGIN {  BEGIN {
 # Get columns of MySQL metadata table      # Get columns of MySQL metadata table
     @columns=&Apache::lonmysql::col_order('metadata');      @columns=&Apache::lonmysql::col_order('metadata');
 }  }
 1;  
 __END__  
   
   
   
   
   
   #
   # Turn MySQL row into hash
   #     This routine should be moved to lonmetadata
   #     a more generic place since it has nothing to do with metadata
   sub metadata_col_to_hash {
       my @cols=@_;
       my %hash=();
       for (my $i=0; $i<=$#columns; $i++) {
    $hash{$columns[$i]}=$cols[$i];
       }
       return %hash;
   }
   
   
   1;
   __END__

Removed from v.1.60  
changed lines
  Added in v.1.67


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