File:  [LON-CAPA] / loncom / interface / lonmeta.pm
Revision 1.107: download - view: text, annotated - select for diffs
Thu Aug 25 20:46:32 2005 UTC (18 years, 9 months ago) by banghart
Branches: MAIN
CVS tags: HEAD
	Change multi pulldown to checkboxes for instructor metadata.

    1: # The LearningOnline Network with CAPA
    2: # Metadata display handler
    3: #
    4: # $Id: lonmeta.pm,v 1.107 2005/08/25 20:46:32 banghart Exp $
    5: #
    6: # Copyright Michigan State University Board of Trustees
    7: #
    8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
    9: #
   10: # LON-CAPA is free software; you can redistribute it and/or modify
   11: # it under the terms of the GNU General Public License as published by
   12: # the Free Software Foundation; either version 2 of the License, or
   13: # (at your option) any later version.
   14: #
   15: # LON-CAPA is distributed in the hope that it will be useful,
   16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
   17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   18: # GNU General Public License for more details.
   19: #
   20: # You should have received a copy of the GNU General Public License 
   21: # along with LON-CAPA; if not, write to the Free Software
   22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
   23: #
   24: # /home/httpd/html/adm/gpl.txt
   25: #
   26: # http://www.lon-capa.org/
   27: 
   28: 
   29: package Apache::lonmeta;
   30: 
   31: use strict;
   32: use LONCAPA::lonmetadata();
   33: use Apache::Constants qw(:common);
   34: use Apache::lonnet;
   35: use Apache::loncommon();
   36: use Apache::lonhtmlcommon(); 
   37: use Apache::lonmsg;
   38: use Apache::lonpublisher;
   39: use Apache::lonlocal;
   40: use Apache::lonmysql;
   41: use Apache::lonmsg;
   42: 
   43: 
   44: ############################################################
   45: ############################################################
   46: ##
   47: ## &get_dynamic_metadata_from_sql($url)
   48: ## 
   49: ## Queries sql database for dynamic metdata
   50: ## Returns a hash of hashes, with keys of urls which match $url
   51: ## Returned fields are given below.
   52: ##
   53: ## Examples:
   54: ## 
   55: ## %DynamicMetadata = &Apache::lonmeta::get_dynmaic_metadata_from_sql
   56: ##     ('/res/msu/korte/');
   57: ##
   58: ## $DynamicMetadata{'/res/msu/korte/example.problem'}->{$field}
   59: ##
   60: ############################################################
   61: ############################################################
   62: sub get_dynamic_metadata_from_sql {
   63:     my ($url) = shift();
   64:     my ($authordom,$author)=($url=~m:^/res/(\w+)/(\w+)/:);
   65:     if (! defined($authordom)) {
   66:         $authordom = shift();
   67:     }
   68:     if  (! defined($author)) { 
   69:         $author = shift();
   70:     }
   71:     if (! defined($authordom) || ! defined($author)) {
   72:         return ();
   73:     }
   74:     my @Fields = ('url','count','course',
   75:                   'goto','goto_list',
   76:                   'comefrom','comefrom_list',
   77:                   'sequsage','sequsage_list',
   78:                   'stdno','stdno_list',
   79: 		  'dependencies',
   80:                   'avetries','avetries_list',
   81:                   'difficulty','difficulty_list',
   82:                   'disc','disc_list',
   83:                   'clear','technical','correct',
   84:                   'helpful','depth');
   85:     #
   86:     my $query = 'SELECT '.join(',',@Fields).
   87:         ' FROM metadata WHERE url LIKE "'.$url.'%"';
   88:     my $server = &Apache::lonnet::homeserver($author,$authordom);
   89:     my $reply = &Apache::lonnet::metadata_query($query,undef,undef,
   90:                                                 ,[$server]);
   91:     return () if (! defined($reply) || ref($reply) ne 'HASH');
   92:     my $filename = $reply->{$server};
   93:     if (! defined($filename) || $filename =~ /^error/) {
   94:         return ();
   95:     }
   96:     my $max_time = time + 10; # wait 10 seconds for results at most
   97:     my %ReturnHash;
   98:     #
   99:     # Look for results
  100:     my $finished = 0;
  101:     while (! $finished && time < $max_time) {
  102:         my $datafile=$Apache::lonnet::perlvar{'lonDaemons'}.'/tmp/'.$filename;
  103:         if (! -e "$datafile.end") { next; }
  104:         my $fh;
  105:         if (!($fh=Apache::File->new($datafile))) { next; }
  106:         while (my $result = <$fh>) {
  107:             chomp($result);
  108:             next if (! $result);
  109:             my @Data = 
  110:                 map { 
  111:                     &Apache::lonnet::unescape($_); 
  112:                 } split(',',$result);
  113:             my $url = $Data[0];
  114:             for (my $i=0;$i<=$#Fields;$i++) {
  115:                 $ReturnHash{$url}->{$Fields[$i]}=$Data[$i];
  116:             }
  117:         }
  118:         $finished = 1;
  119:     }
  120:     #
  121:     return %ReturnHash;
  122: }
  123: 
  124: 
  125: # Fetch and evaluate dynamic metadata
  126: sub dynamicmeta {
  127:     my $url=&Apache::lonnet::declutter(shift);
  128:     $url=~s/\.meta$//;
  129:     my ($adomain,$aauthor)=($url=~/^(\w+)\/(\w+)\//);
  130:     my $regexp=$url;
  131:     $regexp=~s/(\W)/\\$1/g;
  132:     $regexp='___'.$regexp.'___';
  133:     my %evaldata=&Apache::lonnet::dump('nohist_resevaldata',$adomain,
  134: 				       $aauthor,$regexp);
  135:     my %DynamicData = &LONCAPA::lonmetadata::process_reseval_data(\%evaldata);
  136:     my %Data = &LONCAPA::lonmetadata::process_dynamic_metadata($url,
  137:                                                                \%DynamicData);
  138:     #
  139:     # Deal with 'count' separately
  140:     $Data{'count'} = &access_count($url,$aauthor,$adomain);
  141:     #
  142:     # Debugging code I will probably need later
  143:     if (0) {
  144:         &Apache::lonnet::logthis('Dynamic Metadata');
  145:         while(my($k,$v)=each(%Data)){
  146:             &Apache::lonnet::logthis('    "'.$k.'"=>"'.$v.'"');
  147:         }
  148:         &Apache::lonnet::logthis('-------------------');
  149:     }
  150:     return %Data;
  151: }
  152: 
  153: sub access_count {
  154:     my ($src,$author,$adomain) = @_;
  155:     my %countdata=&Apache::lonnet::dump('nohist_accesscount',$adomain,
  156:                                         $author,$src);
  157:     if (! exists($countdata{$src})) {
  158:         return &mt('Not Available');
  159:     } else {
  160:         return $countdata{$src};
  161:     }
  162: }
  163: 
  164: # Try to make an alt tag if there is none
  165: sub alttag {
  166:     my ($base,$src)=@_;
  167:     my $fullpath=&Apache::lonnet::hreflocation($base,$src);
  168:     my $alttag=&Apache::lonnet::metadata($fullpath,'title').' '.
  169:         &Apache::lonnet::metadata($fullpath,'subject').' '.
  170:         &Apache::lonnet::metadata($fullpath,'abstract');
  171:     $alttag=~s/\s+/ /gs;
  172:     $alttag=~s/\"//gs;
  173:     $alttag=~s/\'//gs;
  174:     $alttag=~s/\s+$//gs;
  175:     $alttag=~s/^\s+//gs;
  176:     if ($alttag) { 
  177:         return $alttag; 
  178:     } else { 
  179:         return &mt('No information available'); 
  180:     }
  181: }
  182: 
  183: # Author display
  184: sub authordisplay {
  185:     my ($aname,$adom)=@_;
  186:     return &Apache::loncommon::aboutmewrapper
  187:         (&Apache::loncommon::plainname($aname,$adom),
  188:          $aname,$adom,'preview').' <tt>['.$aname.'@'.$adom.']</tt>';
  189: }
  190: 
  191: # Pretty display
  192: sub evalgraph {
  193:     my $value=shift;
  194:     if (! $value) { 
  195:         return '';
  196:     }
  197:     my $val=int($value*10.+0.5)-10;
  198:     my $output='<table border="0" cellpadding="0" cellspacing="0"><tr>';
  199:     if ($val>=20) {
  200: 	$output.='<td width="20" bgcolor="#555555">&nbsp&nbsp;</td>';
  201:     } else {
  202:         $output.='<td width="'.($val).'" bgcolor="#555555">&nbsp;</td>'.
  203:                  '<td width="'.(20-$val).'" bgcolor="#FF3333">&nbsp;</td>';
  204:     }
  205:     $output.='<td bgcolor="#FFFF33">&nbsp;</td>';
  206:     if ($val>20) {
  207: 	$output.='<td width="'.($val-20).'" bgcolor="#33FF33">&nbsp;</td>'.
  208:                  '<td width="'.(40-$val).'" bgcolor="#555555">&nbsp;</td>';
  209:     } else {
  210:         $output.='<td width="20" bgcolor="#555555">&nbsp&nbsp;</td>';
  211:     }
  212:     $output.='<td> ('.sprintf("%5.2f",$value).') </td></tr></table>';
  213:     return $output;
  214: }
  215: 
  216: sub diffgraph {
  217:     my $value=shift;
  218:     if (! $value) { 
  219:         return '';
  220:     }
  221:     my $val=int(40.0*$value+0.5);
  222:     my @colors=('#FF9933','#EEAA33','#DDBB33','#CCCC33',
  223:                 '#BBDD33','#CCCC33','#DDBB33','#EEAA33');
  224:     my $output='<table border="0" cellpadding="0" cellspacing="0"><tr>';
  225:     for (my $i=0;$i<8;$i++) {
  226: 	if ($val>$i*5) {
  227:             $output.='<td width="5" bgcolor="'.$colors[$i].'">&nbsp;</td>';
  228:         } else {
  229: 	    $output.='<td width="5" bgcolor="#555555">&nbsp;</td>';
  230: 	}
  231:     }
  232:     $output.='<td> ('.sprintf("%3.2f",$value).') </td></tr></table>';
  233:     return $output;
  234: }
  235: 
  236: 
  237: # The field names
  238: sub fieldnames {
  239:     my $file_type=shift;
  240:     my %fields = 
  241:         ('title' => 'Title',
  242:          'author' =>'Author(s)',
  243:          'authorspace' => 'Author Space',
  244:          'modifyinguser' => 'Last Modifying User',
  245:          'subject' => 'Subject',
  246:          'keywords' => 'Keyword(s)',
  247:          'notes' => 'Notes',
  248:          'abstract' => 'Abstract',
  249:          'lowestgradelevel' => 'Lowest Grade Level',
  250:          'highestgradelevel' => 'Highest Grade Level');
  251:     if (! defined($file_type) || $file_type ne 'portfolio') {
  252:         %fields = 
  253:         (%fields,
  254:          'domain' => 'Domain',
  255:          'standards' => 'Standards',
  256:          'mime' => 'MIME Type',
  257:          'language' => 'Language',
  258:          'creationdate' => 'Creation Date',
  259:          'lastrevisiondate' => 'Last Revision Date',
  260:          'owner' => 'Publisher/Owner',
  261:          'copyright' => 'Copyright/Distribution',
  262:          'customdistributionfile' => 'Custom Distribution File',
  263:          'sourceavail' => 'Source Available',
  264:          'sourcerights' => 'Source Custom Distribution File',
  265:          'obsolete' => 'Obsolete',
  266:          'obsoletereplacement' => 'Suggested Replacement for Obsolete File',
  267:          'count'      => 'Network-wide number of accesses (hits)',
  268:          'course'     => 'Network-wide number of courses using resource',
  269:          'course_list' => 'Network-wide courses using resource',
  270:          'sequsage'      => 'Number of resources using or importing resource',
  271:          'sequsage_list' => 'Resources using or importing resource',
  272:          'goto'       => 'Number of resources that follow this resource in maps',
  273:          'goto_list'  => 'Resources that follow this resource in maps',
  274:          'comefrom'   => 'Number of resources that lead up to this resource in maps',
  275:          'comefrom_list' => 'Resources that lead up to this resource in maps',
  276:          'clear'      => 'Material presented in clear way',
  277:          'depth'      => 'Material covered with sufficient depth',
  278:          'helpful'    => 'Material is helpful',
  279:          'correct'    => 'Material appears to be correct',
  280:          'technical'  => 'Resource is technically correct', 
  281:          'avetries'   => 'Average number of tries till solved',
  282:          'stdno'      => 'Total number of students who have worked on this problem',
  283:          'difficulty' => 'Degree of difficulty',
  284:          'disc'       => 'Degree of discrimination',
  285: 	 'dependencies' => 'Resources used by this resource',
  286:          );
  287:     }
  288:     return &Apache::lonlocal::texthash(%fields);
  289: }
  290: 
  291: sub select_course {
  292:     my ($r)=@_;
  293:     $r->print('<h3>Instructor Selected Meta-Data</h3><br />');
  294:     $r->print('<form action="" method="post">');
  295:     $r->print('Select your course<br />');
  296:     $r->print('<select name="metacourse" >');
  297:     my $meta_not_found = 1;
  298:     foreach my $key (keys %env) {        
  299:         if ($key =~ /\.metadata\./) {
  300:             if ($meta_not_found) {
  301:                 undef($meta_not_found);
  302:                 $r->print('<h3>Instructor Selected Meta-Data</h3><br />');
  303:                 $r->print('<form action="" method="post">');
  304:                 $r->print('Select your course<br />');
  305:                 $r->print('<select name="metacourse" >');
  306:             }
  307:             my $course_key = $key;
  308:             $course_key =~ s/\.metadata\..*//;
  309:             $r->print('<option value="'.$course_key.'">');
  310:             $r->print($env{$course_key.'.description'});
  311:             $r->print('</option>');
  312:         }
  313:     }
  314:     unless ($meta_not_found) {
  315:         $r->print('</select><br />');
  316:         $r->print('<input type="submit" value="Assign Instructor Metadata" />');
  317:         $r->print('</form>');
  318:     }
  319:     return 'ok';
  320: }
  321: # Pretty printing of metadata field
  322: 
  323: sub prettyprint {
  324:     my ($type,$value,$target,$prefix,$form,$noformat)=@_;
  325: # $target,$prefix,$form are optional and for filecrumbs only
  326:     if (! defined($value)) { 
  327:         return '&nbsp;'; 
  328:     }
  329:     # Title
  330:     if ($type eq 'title') {
  331: 	return '<font size="+1" face="arial">'.$value.'</font>';
  332:     }
  333:     # Dates
  334:     if (($type eq 'creationdate') ||
  335: 	($type eq 'lastrevisiondate')) {
  336: 	return ($value?&Apache::lonlocal::locallocaltime(
  337: 			  &Apache::lonmysql::unsqltime($value)):
  338: 		&mt('not available'));
  339:     }
  340:     # Language
  341:     if ($type eq 'language') {
  342: 	return &Apache::loncommon::languagedescription($value);
  343:     }
  344:     # Copyright
  345:     if ($type eq 'copyright') {
  346: 	return &Apache::loncommon::copyrightdescription($value);
  347:     }
  348:     # Copyright
  349:     if ($type eq 'sourceavail') {
  350: 	return &Apache::loncommon::source_copyrightdescription($value);
  351:     }
  352:     # MIME
  353:     if ($type eq 'mime') {
  354:         return '<img src="'.&Apache::loncommon::icon($value).'" />&nbsp;'.
  355:             &Apache::loncommon::filedescription($value);
  356:     }
  357:     # Person
  358:     if (($type eq 'author') || 
  359: 	($type eq 'owner') ||
  360: 	($type eq 'modifyinguser') ||
  361: 	($type eq 'authorspace')) {
  362: 	$value=~s/(\w+)(\:|\@)(\w+)/&authordisplay($1,$3)/gse;
  363: 	return $value;
  364:     }
  365:     # Gradelevel
  366:     if (($type eq 'lowestgradelevel') ||
  367: 	($type eq 'highestgradelevel')) {
  368: 	return &Apache::loncommon::gradeleveldescription($value);
  369:     }
  370:     # Only for advance users below
  371:     if (! $env{'user.adv'}) { 
  372:         return '<i>- '.&mt('not displayed').' -</i>';
  373:     }
  374:     # File
  375:     if (($type eq 'customdistributionfile') ||
  376: 	($type eq 'obsoletereplacement') ||
  377: 	($type eq 'goto_list') ||
  378: 	($type eq 'comefrom_list') ||
  379: 	($type eq 'sequsage_list') ||
  380: 	($type eq 'dependencies')) {
  381: 	return '<ul><font size="-1">'.join("\n",map {
  382:             my $url = &Apache::lonnet::clutter($_);
  383:             my $title = &Apache::lonnet::gettitle($url);
  384:             if ($title eq '') {
  385:                 $title = 'Untitled';
  386:                 if ($url =~ /\.sequence$/) {
  387:                     $title .= ' Sequence';
  388:                 } elsif ($url =~ /\.page$/) {
  389:                     $title .= ' Page';
  390:                 } elsif ($url =~ /\.problem$/) {
  391:                     $title .= ' Problem';
  392:                 } elsif ($url =~ /\.html$/) {
  393:                     $title .= ' HTML document';
  394:                 } elsif ($url =~ m:/syllabus$:) {
  395:                     $title .= ' Syllabus';
  396:                 } 
  397:             }
  398:             $_ = '<li>'.$title.' '.
  399: 		&Apache::lonhtmlcommon::crumbs($url,$target,$prefix,$form,'-1',$noformat).
  400:                 '</li>'
  401: 	    } split(/\s*\,\s*/,$value)).'</ul></font>';
  402:     }
  403:     # Evaluations
  404:     if (($type eq 'clear') ||
  405: 	($type eq 'depth') ||
  406: 	($type eq 'helpful') ||
  407: 	($type eq 'correct') ||
  408: 	($type eq 'technical')) {
  409: 	return &evalgraph($value);
  410:     }
  411:     # Difficulty
  412:     if ($type eq 'difficulty' || $type eq 'disc') {
  413: 	return &diffgraph($value);
  414:     }
  415:     # List of courses
  416:     if ($type=~/\_list/) {
  417:         my @Courses = split(/\s*\,\s*/,$value);
  418:         my $Str;
  419:         foreach my $course (@Courses) {
  420:             my %courseinfo = &Apache::lonnet::coursedescription($course);
  421:             if (! exists($courseinfo{'num'}) || $courseinfo{'num'} eq '') {
  422:                 next;
  423:             }
  424:             if ($Str ne '') { $Str .= '<br />'; }
  425:             $Str .= '<a href="/public/'.$courseinfo{'domain'}.'/'.
  426:                 $courseinfo{'num'}.'/syllabus" target="preview">'.
  427:                 $courseinfo{'description'}.'</a>';
  428:         }
  429: 	return $Str;
  430:     }
  431:     # No pretty print found
  432:     return $value;
  433: }
  434: 
  435: # Pretty input of metadata field
  436: sub direct {
  437:     return shift;
  438: }
  439: 
  440: sub selectbox {
  441:     my ($name,$value,$functionref,@idlist)=@_;
  442:     if (! defined($functionref)) {
  443:         $functionref=\&direct;
  444:     }
  445:     my $selout='<select name="'.$name.'">';
  446:     foreach (@idlist) {
  447:         $selout.='<option value=\''.$_.'\'';
  448:         if ($_ eq $value) {
  449: 	    $selout.=' selected>'.&{$functionref}($_).'</option>';
  450: 	}
  451:         else {$selout.='>'.&{$functionref}($_).'</option>';}
  452:     }
  453:     return $selout.'</select>';
  454: }
  455: 
  456: sub relatedfield {
  457:     my ($show,$relatedsearchflag,$relatedsep,$fieldname,$relatedvalue)=@_;
  458:     if (! $relatedsearchflag) { 
  459:         return '';
  460:     }
  461:     if (! defined($relatedsep)) {
  462:         $relatedsep=' ';
  463:     }
  464:     if (! $show) {
  465:         return $relatedsep.'&nbsp;';
  466:     }
  467:     return $relatedsep.'<input type="checkbox" name="'.$fieldname.'_related"'.
  468: 	($relatedvalue?' checked="1"':'').' />';
  469: }
  470: 
  471: sub prettyinput {
  472:     my ($type,$value,$fieldname,$formname,
  473: 	$relatedsearchflag,$relatedsep,$relatedvalue,$size)=@_;
  474:     if (! defined($size)) {
  475:         $size = 80;
  476:     }
  477:     # Language
  478:     if ($type eq 'language') {
  479: 	return &selectbox($fieldname,
  480: 			  $value,
  481: 			  \&Apache::loncommon::languagedescription,
  482: 			  (&Apache::loncommon::languageids)).
  483:                               &relatedfield(0,$relatedsearchflag,$relatedsep);
  484:     }
  485:     # Copyright
  486:     if ($type eq 'copyright') {
  487: 	return &selectbox($fieldname,
  488: 			  $value,
  489: 			  \&Apache::loncommon::copyrightdescription,
  490: 			  (&Apache::loncommon::copyrightids)).
  491:                               &relatedfield(0,$relatedsearchflag,$relatedsep);
  492:     }
  493:     # Source Copyright
  494:     if ($type eq 'sourceavail') {
  495: 	return &selectbox($fieldname,
  496: 			  $value,
  497: 			  \&Apache::loncommon::source_copyrightdescription,
  498: 			  (&Apache::loncommon::source_copyrightids)).
  499:                               &relatedfield(0,$relatedsearchflag,$relatedsep);
  500:     }
  501:     # Gradelevels
  502:     if (($type eq 'lowestgradelevel') ||
  503: 	($type eq 'highestgradelevel')) {
  504: 	return &Apache::loncommon::select_level_form($value,$fieldname).
  505:             &relatedfield(0,$relatedsearchflag,$relatedsep);
  506:     }
  507:     # Obsolete
  508:     if ($type eq 'obsolete') {
  509: 	return '<input type="checkbox" name="'.$fieldname.'"'.
  510: 	    ($value?' checked="1"':'').' />'.
  511:             &relatedfield(0,$relatedsearchflag,$relatedsep); 
  512:     }
  513:     # Obsolete replacement file
  514:     if ($type eq 'obsoletereplacement') {
  515: 	return '<input type="text" name="'.$fieldname.
  516: 	    '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.
  517: 	    "('".$formname."','".$fieldname."'".
  518: 	    ",'')\">".&mt('Select').'</a>'.
  519:             &relatedfield(0,$relatedsearchflag,$relatedsep); 
  520:     }
  521:     # Customdistribution file
  522:     if ($type eq 'customdistributionfile') {
  523: 	return '<input type="text" name="'.$fieldname.
  524: 	    '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.
  525: 	    "('".$formname."','".$fieldname."'".
  526: 	    ",'rights')\">".&mt('Select').'</a>'.
  527:             &relatedfield(0,$relatedsearchflag,$relatedsep); 
  528:     }
  529:     # Source Customdistribution file
  530:     if ($type eq 'sourcerights') {
  531: 	return '<input type="text" name="'.$fieldname.
  532: 	    '" size="60" value="'.$value.'" /><a href="javascript:openbrowser'.
  533: 	    "('".$formname."','".$fieldname."'".
  534: 	    ",'rights')\">".&mt('Select').'</a>'.
  535:             &relatedfield(0,$relatedsearchflag,$relatedsep); 
  536:     }
  537:     # Dates
  538:     if (($type eq 'creationdate') ||
  539: 	($type eq 'lastrevisiondate')) {
  540: 	return 
  541:             &Apache::lonhtmlcommon::date_setter($formname,$fieldname,$value).
  542:             &relatedfield(0,$relatedsearchflag,$relatedsep);
  543:     }
  544:     # No pretty input found
  545:     $value=~s/^\s+//gs;
  546:     $value=~s/\s+$//gs;
  547:     $value=~s/\s+/ /gs;
  548:     $value=~s/\"/\&quot\;/gs;
  549:     return 
  550:         '<input type="text" name="'.$fieldname.'" size="'.$size.'" '.
  551:         'value="'.$value.'" />'.
  552:         &relatedfield(1,$relatedsearchflag,$relatedsep,$fieldname,
  553:                       $relatedvalue); 
  554: }
  555: 
  556: # Main Handler
  557: sub handler {
  558:     my $r=shift;
  559:     #
  560:     my $uri=$r->uri;
  561:     #
  562:     # Set document type
  563:     &Apache::loncommon::content_type($r,'text/html');
  564:     $r->send_http_header;
  565:     return OK if $r->header_only;
  566:     #
  567:     my ($resdomain,$resuser)=
  568:         (&Apache::lonnet::declutter($uri)=~/^(\w+)\/(\w+)\//);
  569:     my $html=&Apache::lonxml::xmlbegin();
  570:     $r->print($html.'<head><title>'.
  571:               'Catalog Information'.
  572:               '</title></head>');
  573:     if ($uri=~m:/adm/bombs/(.*)$:) {
  574:         $r->print(&Apache::loncommon::bodytag('Error Messages'));
  575:         # Looking for all bombs?
  576:         &report_bombs($r,$uri);
  577:     } elsif ($uri=~/\/portfolio\//) {
  578:         $r->print(&Apache::loncommon::bodytag
  579:           ('Edit Portfolio File Information','','','',$resdomain));
  580:         &present_editable_metadata($r,$uri,'portfolio');
  581:         &select_course($r);
  582:     } elsif ($uri=~/^\/\~/) { 
  583:         # Construction space
  584:         $r->print(&Apache::loncommon::bodytag
  585:                   ('Edit Catalog Information','','','',$resdomain));
  586:         &present_editable_metadata($r,$uri);
  587:     } else {
  588:         $r->print(&Apache::loncommon::bodytag
  589: 		  ('Catalog Information','','','',$resdomain));
  590:         &present_uneditable_metadata($r,$uri);
  591:     }
  592:     $r->print('</body></html>');
  593:     return OK;
  594: }
  595: 
  596: #####################################################
  597: #####################################################
  598: ###                                               ###
  599: ###                Report Bombs                   ###
  600: ###                                               ###
  601: #####################################################
  602: #####################################################
  603: sub report_bombs {
  604:     my ($r,$uri) = @_;
  605:     # Set document type
  606:     $uri =~ s:/adm/bombs/::;
  607:     $uri = &Apache::lonnet::declutter($uri);
  608:     $r->print('<h1>'.&Apache::lonnet::clutter($uri).'</h1>');
  609:     my ($domain,$author)=($uri=~/^(\w+)\/(\w+)\//);
  610:     if (&Apache::loncacc::constructaccess('/~'.$author.'/',$domain)) {
  611: 	if ($env{'form.clearbombs'}) {
  612: 	    &Apache::lonmsg::clear_author_res_msg($uri);
  613: 	}
  614:         my $clear=&mt('Clear all Messages in Subdirectory');
  615: 	$r->print(<<ENDCLEAR);
  616: <form method="post">
  617: <input type="submit" name="clearbombs" value="$clear" />
  618: </form>
  619: ENDCLEAR
  620:         my %brokenurls = 
  621:             &Apache::lonmsg::all_url_author_res_msg($author,$domain);
  622:         foreach (sort(keys(%brokenurls))) {
  623:             if ($_=~/^\Q$uri\E/) {
  624:                 $r->print
  625:                     ('<a href="'.&Apache::lonnet::clutter($_).'">'.$_.'</a>'.
  626:                      &Apache::lonmsg::retrieve_author_res_msg($_).
  627:                      '<hr />');
  628:             }
  629:         }
  630:     } else {
  631:         $r->print(&mt('Not authorized'));
  632:     }
  633:     return;
  634: }
  635: 
  636: #####################################################
  637: #####################################################
  638: ###                                               ###
  639: ###        Uneditable Metadata Display            ###
  640: ###                                               ###
  641: #####################################################
  642: #####################################################
  643: sub present_uneditable_metadata {
  644:     my ($r,$uri) = @_;
  645:     #
  646:     my %content=();
  647:     # Read file
  648:     foreach (split(/\,/,&Apache::lonnet::metadata($uri,'keys'))) {
  649:         $content{$_}=&Apache::lonnet::metadata($uri,$_);
  650:     }
  651:     # Render Output
  652:     # displayed url
  653:     my ($thisversion)=($uri=~/\.(\d+)\.(\w+)\.meta$/);
  654:     $uri=~s/\.meta$//;
  655:     my $disuri=&Apache::lonnet::clutter($uri);
  656:     # version
  657:     my $currentversion=&Apache::lonnet::getversion($disuri);
  658:     my $versiondisplay='';
  659:     if ($thisversion) {
  660:         $versiondisplay=&mt('Version').': '.$thisversion.
  661:             ' ('.&mt('most recent version').': '.
  662:             ($currentversion>0 ? 
  663:              $currentversion   :
  664:              &mt('information not available')).')';
  665:     } else {
  666:         $versiondisplay='Version: '.$currentversion;
  667:     }
  668:     # crumbify displayed URL               uri     target prefix form  size
  669:     $disuri=&Apache::lonhtmlcommon::crumbs($disuri,undef, undef, undef,'+1');
  670:     $disuri =~ s:<br />::g;
  671:     # obsolete
  672:     my $obsolete=$content{'obsolete'};
  673:     my $obsoletewarning='';
  674:     if (($obsolete) && ($env{'user.adv'})) {
  675:         $obsoletewarning='<p><font color="red">'.
  676:             &mt('This resource has been marked obsolete by the author(s)').
  677:             '</font></p>';
  678:     }
  679:     #
  680:     my %lt=&fieldnames();
  681:     my $table='';
  682:     my $title = $content{'title'};
  683:     if (! defined($title)) {
  684:         $title = 'Untitled Resource';
  685:     }
  686:     foreach ('title', 
  687:              'author', 
  688:              'subject', 
  689:              'keywords', 
  690:              'notes', 
  691:              'abstract',
  692:              'lowestgradelevel',
  693:              'highestgradelevel',
  694:              'standards', 
  695:              'mime', 
  696:              'language', 
  697:              'creationdate', 
  698:              'lastrevisiondate', 
  699:              'owner', 
  700:              'copyright', 
  701:              'customdistributionfile',
  702:              'sourceavail',
  703:              'sourcerights', 
  704:              'obsolete', 
  705:              'obsoletereplacement') {
  706:         $table.='<tr><td bgcolor="#AAAAAA">'.$lt{$_}.
  707:             '</td><td bgcolor="#CCCCCC">'.
  708:             &prettyprint($_,$content{$_}).'</td></tr>';
  709:         delete $content{$_};
  710:     }
  711:     #
  712:     $r->print(<<ENDHEAD);
  713: <h2>$title</h2>
  714: <p>
  715: $disuri<br />
  716: $obsoletewarning
  717: $versiondisplay
  718: </p>
  719: <table cellspacing="2" border="0">
  720: $table
  721: </table>
  722: ENDHEAD
  723:     if ($env{'user.adv'}) {
  724:         &print_dynamic_metadata($r,$uri,\%content);
  725:     }
  726:     return;
  727: }
  728: 
  729: sub print_dynamic_metadata {
  730:     my ($r,$uri,$content) = @_;
  731:     #
  732:     my %content = %$content;
  733:     my %lt=&fieldnames();
  734:     #
  735:     my $description = 'Dynamic Metadata (updated periodically)';
  736:     $r->print('<h3>'.&mt($description).'</h3>'.
  737:               &mt('Processing'));
  738:     $r->rflush();
  739:     my %items=&fieldnames();
  740:     my %dynmeta=&dynamicmeta($uri);
  741:     #
  742:     # General Access and Usage Statistics
  743:     if (exists($dynmeta{'count'}) ||
  744:         exists($dynmeta{'sequsage'}) ||
  745:         exists($dynmeta{'comefrom'}) ||
  746:         exists($dynmeta{'goto'}) ||
  747:         exists($dynmeta{'course'})) {
  748:         $r->print('<h4>'.&mt('Access and Usage Statistics').'</h4>'.
  749:                   '<table cellspacing="2" border="0">');
  750:         foreach ('count',
  751:                  'sequsage','sequsage_list',
  752:                  'comefrom','comefrom_list',
  753:                  'goto','goto_list',
  754:                  'course','course_list') {
  755:             $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
  756:                       '<td bgcolor="#CCCCCC">'.
  757:                       &prettyprint($_,$dynmeta{$_})."</td></tr>\n");
  758:         }
  759:         $r->print('</table>');
  760:     } else {
  761:         $r->print('<h4>'.&mt('No Access or Usages Statistics are available for this resource.').'</h4>');
  762:     }
  763:     #
  764:     # Assessment statistics
  765:     if ($uri=~/\.(problem|exam|quiz|assess|survey|form)$/) {
  766:         if (exists($dynmeta{'stdno'}) ||
  767:             exists($dynmeta{'avetries'}) ||
  768:             exists($dynmeta{'difficulty'}) ||
  769:             exists($dynmeta{'disc'})) {
  770:             # This is an assessment, print assessment data
  771:             $r->print('<h4>'.
  772:                       &mt('Overall Assessment Statistical Data').
  773:                       '</h4>'.
  774:                       '<table cellspacing="2" border="0">');
  775:             $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{'stdno'}.'</td>'.
  776:                       '<td bgcolor="#CCCCCC">'.
  777:                       &prettyprint('stdno',$dynmeta{'stdno'}).
  778:                       '</td>'."</tr>\n");
  779:             foreach ('avetries','difficulty','disc') {
  780:                 $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
  781:                           '<td bgcolor="#CCCCCC">'.
  782:                           &prettyprint($_,sprintf('%5.2f',$dynmeta{$_})).
  783:                           '</td>'."</tr>\n");
  784:             }
  785:             $r->print('</table>');    
  786:         }
  787:         if (exists($dynmeta{'stats'})) {
  788:             #
  789:             # New assessment statistics
  790:             $r->print('<h4>'.
  791:                       &mt('Detailed Assessment Statistical Data').
  792:                       '</h4>');
  793:             my $table = '<table cellspacing="2" border="0">'.
  794:                 '<tr>'.
  795:                 '<th>Course</th>'.
  796:                 '<th>Section(s)</th>'.
  797:                 '<th>Num Students</th>'.
  798:                 '<th>Mean Tries</th>'.
  799:                 '<th>Degree of Difficulty</th>'.
  800:                 '<th>Degree of Discrimination</th>'.
  801:                 '<th>Time of computation</th>'.
  802:                 '</tr>'.$/;
  803:             foreach my $identifier (sort(keys(%{$dynmeta{'stats'}}))) {
  804:                 my $data = $dynmeta{'stats'}->{$identifier};
  805:                 my $course = $data->{'course'};
  806:                 my %courseinfo = &Apache::lonnet::coursedescription($course);
  807:                 if (! exists($courseinfo{'num'}) || $courseinfo{'num'} eq '') {
  808:                     &Apache::lonnet::logthis('lookup for '.$course.' failed');
  809:                     next;
  810:                 }
  811:                 $table .= '<tr>';
  812:                 $table .= 
  813:                     '<td><nobr>'.$courseinfo{'description'}.'</nobr></td>';
  814:                 $table .= 
  815:                     '<td align="right">'.$data->{'sections'}.'</td>';
  816:                 $table .=
  817:                     '<td align="right">'.$data->{'stdno'}.'</td>';
  818:                 foreach ('avetries','difficulty','disc') {
  819:                     $table .= '<td align="right">';
  820:                     if (exists($data->{$_})) {
  821:                         $table .= sprintf('%.2f',$data->{$_}).'&nbsp;';
  822:                     } else {
  823:                         $table .= '';
  824:                     }
  825:                     $table .= '</td>';
  826:                 }
  827:                 $table .=
  828:                     '<td><nobr>'.
  829:                     &Apache::lonlocal::locallocaltime($data->{'timestamp'}).
  830:                     '</nobr></td>';
  831:                 $table .=
  832:                     '</tr>'.$/;
  833:             }
  834:             $table .= '</table>'.$/;
  835:             $r->print($table);
  836:         } else {
  837:             $r->print('No new dynamic data found.');
  838:         }
  839:     } else {
  840:         $r->print('<h4>'.
  841:           &mt('No Assessment Statistical Data is available for this resource').
  842:                   '</h4>');
  843:     }
  844: 
  845:     #
  846:     #
  847:     if (exists($dynmeta{'clear'})   || 
  848:         exists($dynmeta{'depth'})   || 
  849:         exists($dynmeta{'helpful'}) || 
  850:         exists($dynmeta{'correct'}) || 
  851:         exists($dynmeta{'technical'})){ 
  852:         $r->print('<h4>'.&mt('Evaluation Data').'</h4>'.
  853:                   '<table cellspacing="2" border="0">');
  854:         foreach ('clear','depth','helpful','correct','technical') {
  855:             $r->print('<tr><td bgcolor="#AAAAAA">'.$lt{$_}.'</td>'.
  856:                       '<td bgcolor="#CCCCCC">'.
  857:                       &prettyprint($_,$dynmeta{$_})."</td></tr>\n");
  858:         }
  859:         $r->print('</table>');
  860:     } else {
  861:         $r->print('<h4>'.&mt('No Evaluation Data is available for this resource.').'</h4>');
  862:     }
  863:     $uri=~/^\/res\/(\w+)\/(\w+)\//; 
  864:     if ((($env{'user.domain'} eq $1) && ($env{'user.name'} eq $2))
  865:         || ($env{'user.role.ca./'.$1.'/'.$2})) {
  866:         if (exists($dynmeta{'comments'})) {
  867:             $r->print('<h4>'.&mt('Evaluation Comments').' ('.
  868:                       &mt('visible to author and co-authors only').
  869:                       ')</h4>'.
  870:                       '<blockquote>'.$dynmeta{'comments'}.'</blockquote>');
  871:         } else {
  872:             $r->print('<h4>'.&mt('There are no Evaluation Comments on this resource.').'</h4>');
  873:         }
  874:         my $bombs = &Apache::lonmsg::retrieve_author_res_msg($uri);
  875:         if (defined($bombs) && $bombs ne '') {
  876:             $r->print('<a name="bombs" /><h4>'.&mt('Error Messages').' ('.
  877:                       &mt('visible to author and co-authors only').')'.
  878:                       '</h4>'.$bombs);
  879:         } else {
  880:             $r->print('<h4>'.&mt('There are currently no Error Messages for this resource.').'</h4>');
  881:         }
  882:     }
  883:     #
  884:     # All other stuff
  885:     $r->print('<h3>'.
  886:               &mt('Additional Metadata (non-standard, parameters, exports)').
  887:               '</h3><table border="0" cellspacing="1">');
  888:     foreach (sort(keys(%content))) {
  889:         my $name=$_;
  890:         if ($name!~/\.display$/) {
  891:             my $display=&Apache::lonnet::metadata($uri,
  892:                                                   $name.'.display');
  893:             if (! $display) { 
  894:                 $display=$name;
  895:             };
  896:             my $otherinfo='';
  897:             foreach ('name','part','type','default') {
  898:                 if (defined(&Apache::lonnet::metadata($uri,
  899:                                                       $name.'.'.$_))) {
  900:                     $otherinfo.=' '.$_.'='.
  901:                         &Apache::lonnet::metadata($uri,
  902:                                                   $name.'.'.$_).'; ';
  903:                 }
  904:             }
  905:             $r->print('<tr><td bgcolor="#bbccbb"><font size="-1" color="#556655">'.$display.'</font></td><td bgcolor="#ccddcc"><font size="-1" color="#556655">'.$content{$name});
  906:             if ($otherinfo) {
  907:                 $r->print(' ('.$otherinfo.')');
  908:             }
  909:             $r->print("</font></td></tr>\n");
  910:         }
  911:     }
  912:     $r->print("</table>");
  913:     return;
  914: }
  915: 
  916: 
  917: 
  918: #####################################################
  919: #####################################################
  920: ###                                               ###
  921: ###          Editable metadata display            ###
  922: ###                                               ###
  923: #####################################################
  924: #####################################################
  925: sub present_editable_metadata {
  926:     my ($r,$uri, $file_type) = @_;
  927:     # Construction Space Call
  928:     # Header
  929:     my $disuri=$uri;
  930:     my $fn=&Apache::lonnet::filelocation('',$uri);
  931:     $disuri=~s/^\/\~/\/priv\//;
  932:     $disuri=~s/\.meta$//;
  933:     my $target=$uri;
  934:     $target=~s/^\/\~/\/res\/$env{'request.role.domain'}\//;
  935:     $target=~s/\.meta$//;
  936:     my $bombs=&Apache::lonmsg::retrieve_author_res_msg($target);
  937:     if ($bombs) {
  938:         my $showdel=1;
  939:         if ($env{'form.delmsg'}) {
  940:             if (&Apache::lonmsg::del_url_author_res_msg($target) eq 'ok') {
  941:                 $bombs=&mt('Messages deleted.');
  942: 		$showdel=0;
  943:             } else {
  944:                 $bombs=&mt('Error deleting messages');
  945:             }
  946:         }
  947:         if ($env{'form.clearmsg'}) {
  948: 	    my $cleardir=$target;
  949: 	    $cleardir=~s/\/[^\/]+$/\//;
  950:             if (&Apache::lonmsg::clear_author_res_msg($cleardir) eq 'ok') {
  951:                 $bombs=&mt('Messages cleared.');
  952: 		$showdel=0;
  953:             } else {
  954:                 $bombs=&mt('Error clearing messages');
  955:             }
  956:         }
  957:         my $del=&mt('Delete Messages for this Resource');
  958: 	my $clear=&mt('Clear all Messages in Subdirectory');
  959: 	my $goback=&mt('Back to Source File');
  960:         $r->print(<<ENDBOMBS);
  961: <h1>$disuri</h1>
  962: <form method="post" name="defaultmeta">
  963: ENDBOMBS
  964:         if ($showdel) {
  965: 	    $r->print(<<ENDDEL);
  966: <input type="submit" name="delmsg" value="$del" />
  967: <input type="submit" name="clearmsg" value="$clear" />
  968: ENDDEL
  969:         } else {
  970:             $r->print('<a href="'.$disuri.'" />'.$goback.'</a>');
  971: 	}
  972: 	$r->print('<br />'.$bombs);
  973:     } else {
  974:         my $displayfile='Catalog Information for '.$disuri;
  975:         if ($disuri=~/\/default$/) {
  976:             my $dir=$disuri;
  977:             $dir=~s/default$//;
  978:             $displayfile=
  979:                 &mt('Default Cataloging Information for Directory').' '.
  980:                 $dir;
  981:         }
  982:         %Apache::lonpublisher::metadatafields=();
  983:         %Apache::lonpublisher::metadatakeys=();
  984:         my $result=&Apache::lonnet::getfile($fn);
  985:         if ($result == -1){
  986:             $r->print('Creating new '.$fn);
  987:         } else {
  988:             &Apache::lonpublisher::metaeval($result);
  989:         }
  990:         $r->print(<<ENDEDIT);
  991: <h1>$displayfile</h1>
  992: <form method="post" name="defaultmeta">
  993: ENDEDIT
  994:         $r->print('<script language="JavaScript">'.
  995:                   &Apache::loncommon::browser_and_searcher_javascript().
  996:                   '</script>');
  997:         my %lt=&fieldnames($file_type);
  998: 	my $output;
  999: 	my @fields;
 1000: 	if ($file_type eq 'portfolio') {
 1001: 	    @fields =  ('author','title','subject','keywords','abstract','notes','lowestgradelevel',
 1002: 	                'highestgradelevel');
 1003: 	} else {
 1004: 	    @fields = ('author','title','subject','keywords','abstract','notes',
 1005:                  'copyright','customdistributionfile','language',
 1006:                  'standards',
 1007:                  'lowestgradelevel','highestgradelevel','sourceavail','sourcerights',
 1008:                  'obsolete','obsoletereplacement');
 1009:         }
 1010:         foreach (@fields) {
 1011:             if (defined($env{'form.new_'.$_})) {
 1012:                 $Apache::lonpublisher::metadatafields{$_}=
 1013:                     $env{'form.new_'.$_};
 1014:             }
 1015:             if (! $Apache::lonpublisher::metadatafields{'copyright'}) {
 1016:                 $Apache::lonpublisher::metadatafields{'copyright'}=
 1017:                     'default';
 1018:             }
 1019:             $output.=('<p>'.$lt{$_}.': '.
 1020:                       &prettyinput($_,
 1021: 				   $Apache::lonpublisher::metadatafields{$_},
 1022: 				   'new_'.$_,'defaultmeta').'</p>');
 1023:             if ($env{'form.metacourse'}) {
 1024:    
 1025:     $r->print('This is the instructor metadata area<br />');
 1026:     my @keywords = sort(split /, /,$env{$env{'form.metacourse'}.'.metadata.keywords'});
 1027:     $r->print($env{'form.metacourse'}.'<br />');
 1028:     $r->print('<form method="post" action="" size="3" >');
 1029:     foreach my $word (@keywords) {
 1030:         my $checked;
 1031:         if ($Apache::lonpublisher::metadatafields{'coursekeyword'}=~ m/$word/) {
 1032:             $checked = 1;
 1033:         } else {
 1034:             undef($checked);
 1035:         }
 1036:         $r->print(&Apache::lonhtmlcommon::checkbox('keywords',$checked,$word).$word.'<br />');
 1037:     }
 1038:     $r->print('<br /><input type="submit" name="store" value="Assign Meta-data" />');
 1039:     $r->print('</form>');
 1040:     return 'ok';
 1041:             }
 1042:         }
 1043:         if ($env{'form.store'}) {
 1044:             my $mfh;
 1045:             my $formname='store'; 
 1046:             my $file_content;
 1047:             if (&Apache::loncommon::get_env_multiple('form.keywords')) {
 1048:             $Apache::lonpublisher::metadatafields{'coursekeyword'} = 
 1049:                         join (', ', &Apache::loncommon::get_env_multiple('form.keywords'));
 1050:             }
 1051:             foreach (sort keys %Apache::lonpublisher::metadatafields) {
 1052:                 next if ($_ =~ /\./);
 1053:                 my $unikey=$_;
 1054:                 $unikey=~/^([A-Za-z]+)/;
 1055:                 my $tag=$1;
 1056:                 $tag=~tr/A-Z/a-z/;
 1057:                 $file_content.= "\n\<$tag";
 1058:                 foreach (split(/\,/,
 1059:                              $Apache::lonpublisher::metadatakeys{$unikey})
 1060:                          ) {
 1061:                     my $value=
 1062:                      $Apache::lonpublisher::metadatafields{$unikey.'.'.$_};
 1063:                     $value=~s/\"/\'\'/g;
 1064:                     $file_content.=' '.$_.'="'.$value.'"' ;
 1065:                     # print $mfh ' '.$_.'="'.$value.'"';
 1066:                 }
 1067:                 $file_content.= '>'.
 1068:                     &HTML::Entities::encode
 1069:                     ($Apache::lonpublisher::metadatafields{$unikey},
 1070:                      '<>&"').
 1071:                      '</'.$tag.'>';
 1072:             }
 1073:             if ($fn =~ /\/portfolio\//) {
 1074:                 $fn =~ /\/portfolio\/(.*)$/;
 1075:                 my $new_fn = '/'.$1;
 1076:                 $env{'form.'.$formname}=$file_content;
 1077:                 $env{'form.'.$formname.'.filename'}=$new_fn;
 1078:                 &Apache::lonnet::userfileupload('uploaddoc','',
 1079: 	        	 'portfolio'.$env{'form.currentpath'});
 1080: 	        my $status =&Apache::lonnet::userfileupload($formname,'','portfolio');
 1081:                 if (&Apache::lonnet::userfileupload($formname,'','portfolio') eq 'error: no uploaded file') {
 1082:                     $r->print('<p><font color="red">'.
 1083:                       &mt('Could not write metadata').', '.
 1084:                      &mt('FAIL').'</font></p>');
 1085:                 } else {
 1086:                     $r->print('<p><font color="blue">'.&mt('Wrote Metadata').
 1087: 		  ' '.&Apache::lonlocal::locallocaltime(time).
 1088: 		  '</font></p>');
 1089:                 }
 1090:             } else {
 1091:                 if (!  ($mfh=Apache::File->new('>'.$fn))) {
 1092:                     $r->print('<p><font color="red">'.
 1093:                         &mt('Could not write metadata').', '.
 1094:                         &mt('FAIL').'</font></p>');
 1095:                 } else {
 1096:                     print $mfh $file_content;
 1097: 		    $r->print('<p><font color="blue">'.&mt('Wrote Metadata').
 1098: 			      ' '.&Apache::lonlocal::locallocaltime(time).
 1099: 			      '</font></p>');
 1100:                 }
 1101:             }
 1102:         }
 1103: 	$r->print($output.'<br /><input type="submit" name="store" value="'.
 1104:                   &mt('Store Catalog Information').'">');
 1105:     }
 1106:     $r->print('</form>');
 1107:     return;
 1108: }
 1109: 
 1110: 1;
 1111: __END__
 1112: 
 1113:      

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