--- loncom/lonnet/perl/lonnet.pm 2022/10/07 12:53:32 1.1493 +++ loncom/lonnet/perl/lonnet.pm 2023/03/27 21:32:41 1.1506 @@ -1,7 +1,7 @@ # The LearningOnline Network # TCP networking package # -# $Id: lonnet.pm,v 1.1493 2022/10/07 12:53:32 raeburn Exp $ +# $Id: lonnet.pm,v 1.1506 2023/03/27 21:32:41 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -985,9 +985,9 @@ sub spareserver { : $userloadpercent; my ($uint_dom,$remotesessions); if (($udom ne '') && (&domain($udom) ne '')) { - my $uprimary_id = &Apache::lonnet::domain($udom,'primary'); - $uint_dom = &Apache::lonnet::internet_dom($uprimary_id); - my %udomdefaults = &Apache::lonnet::get_domain_defaults($udom); + my $uprimary_id = &domain($udom,'primary'); + $uint_dom = &internet_dom($uprimary_id); + my %udomdefaults = &get_domain_defaults($udom); $remotesessions = $udomdefaults{'remotesessions'}; } my $spareshash = &this_host_spares($udom); @@ -1023,7 +1023,7 @@ sub spareserver { if ($protocol{$spare_server} eq 'https') { $protocol = $protocol{$spare_server}; } - my $alias = &Apache::lonnet::use_proxy_alias($r,$spare_server); + my $alias = &use_proxy_alias($r,$spare_server); $hostname = $alias if ($alias ne ''); $spare_server = $protocol.'://'.$hostname; } @@ -1211,7 +1211,7 @@ sub choose_server { unless (defined($cached)) { my $cachetime = 60*60*24; my %domconfig = - &Apache::lonnet::get_dom('configuration',['loadbalancing'],$udom); + &get_dom('configuration',['loadbalancing'],$udom); if (ref($domconfig{'loadbalancing'}) eq 'HASH') { $balancers = &do_cache_new('loadbalancing',$udom,$domconfig{'loadbalancing'}, $cachetime); @@ -1404,7 +1404,7 @@ sub can_switchserver { sub can_host_session { my ($udom,$lonhost,$remoterev,$remotesessions,$hostedsessions) = @_; my $canhost = 1; - my $host_idn = &Apache::lonnet::internet_dom($lonhost); + my $host_idn = &internet_dom($lonhost); if (ref($remotesessions) eq 'HASH') { if (ref($remotesessions->{'excludedomain'}) eq 'ARRAY') { if (grep(/^\Q$host_idn\E$/,@{$remotesessions->{'excludedomain'}})) { @@ -1440,8 +1440,8 @@ sub can_host_session { } if ($canhost) { if (ref($hostedsessions) eq 'HASH') { - my $uprimary_id = &Apache::lonnet::domain($udom,'primary'); - my $uint_dom = &Apache::lonnet::internet_dom($uprimary_id); + my $uprimary_id = &domain($udom,'primary'); + my $uint_dom = &internet_dom($uprimary_id); if (ref($hostedsessions->{'excludedomain'}) eq 'ARRAY') { if (($uint_dom ne '') && (grep(/^\Q$uint_dom\E$/,@{$hostedsessions->{'excludedomain'}}))) { @@ -1533,7 +1533,7 @@ sub spares_for_offload { } else { my $cachetime = 60*60*24; my %domconfig = - &Apache::lonnet::get_dom('configuration',['usersessions'],$dom_in_use); + &get_dom('configuration',['usersessions'],$dom_in_use); if (ref($domconfig{'usersessions'}) eq 'HASH') { if (ref($domconfig{'usersessions'}{'spares'}) eq 'HASH') { if (ref($domconfig{'usersessions'}{'spares'}{$lonhost_in_use}) eq 'HASH') { @@ -1585,9 +1585,9 @@ sub check_loadbalancing { $rule_in_effect,$offloadto,$otherserver,$setcookie,$dom_balancers); my $lonhost = $perlvar{'lonHostID'}; my @hosts = ¤t_machine_ids(); - my $uprimary_id = &Apache::lonnet::domain($udom,'primary'); - my $uintdom = &Apache::lonnet::internet_dom($uprimary_id); - my $intdom = &Apache::lonnet::internet_dom($lonhost); + my $uprimary_id = &domain($udom,'primary'); + my $uintdom = &internet_dom($uprimary_id); + my $intdom = &internet_dom($lonhost); my $serverhomedom = &host_domain($lonhost); my $domneedscache; my $cachetime = 60*60*24; @@ -1601,7 +1601,7 @@ sub check_loadbalancing { my ($result,$cached)=&is_cached_new('loadbalancing',$dom_in_use); unless (defined($cached)) { my %domconfig = - &Apache::lonnet::get_dom('configuration',['loadbalancing'],$dom_in_use); + &get_dom('configuration',['loadbalancing'],$dom_in_use); if (ref($domconfig{'loadbalancing'}) eq 'HASH') { $result = &do_cache_new('loadbalancing',$dom_in_use,$domconfig{'loadbalancing'},$cachetime); } else { @@ -1662,7 +1662,7 @@ sub check_loadbalancing { ($result,$cached)=&is_cached_new('loadbalancing',$serverhomedom); unless (defined($cached)) { my %domconfig = - &Apache::lonnet::get_dom('configuration',['loadbalancing'],$serverhomedom); + &get_dom('configuration',['loadbalancing'],$serverhomedom); if (ref($domconfig{'loadbalancing'}) eq 'HASH') { $result = &do_cache_new('loadbalancing',$serverhomedom,$domconfig{'loadbalancing'},$cachetime); } else { @@ -1795,7 +1795,7 @@ sub get_loadbalancer_targets { } } elsif ($rule_in_effect eq 'externalbalancer') { my %domconfig = - &Apache::lonnet::get_dom('configuration',['loadbalancing'],$udom); + &get_dom('configuration',['loadbalancing'],$udom); if (ref($domconfig{'loadbalancing'}) eq 'HASH') { if ($domconfig{'loadbalancing'}{'lonhost'} ne '') { if (&hostname($domconfig{'loadbalancing'}{'lonhost'}) ne '') { @@ -1859,15 +1859,15 @@ sub trusted_domains { return ($trusted,$untrusted); } my $callprimary = &domain($calldom,'primary'); - my $intcalldom = &Apache::lonnet::internet_dom($callprimary); + my $intcalldom = &internet_dom($callprimary); if ($intcalldom eq '') { return ($trusted,$untrusted); } - my ($trustconfig,$cached)=&Apache::lonnet::is_cached_new('trust',$calldom); + my ($trustconfig,$cached)=&is_cached_new('trust',$calldom); unless (defined($cached)) { - my %domconfig = &Apache::lonnet::get_dom('configuration',['trust'],$calldom); - &Apache::lonnet::do_cache_new('trust',$calldom,$domconfig{'trust'},3600); + my %domconfig = &get_dom('configuration',['trust'],$calldom); + &do_cache_new('trust',$calldom,$domconfig{'trust'},3600); $trustconfig = $domconfig{'trust'}; } if (ref($trustconfig)) { @@ -2345,7 +2345,7 @@ sub get_domainconfiguser { sub retrieve_inst_usertypes { my ($udom) = @_; my (%returnhash,@order); - my %domdefs = &Apache::lonnet::get_domain_defaults($udom); + my %domdefs = &get_domain_defaults($udom); if ((ref($domdefs{'inststatustypes'}) eq 'HASH') && (ref($domdefs{'inststatusorder'}) eq 'ARRAY')) { return ($domdefs{'inststatustypes'},$domdefs{'inststatusorder'}); @@ -2693,12 +2693,13 @@ sub get_domain_defaults { } my %domdefaults; my %domconfig = - &Apache::lonnet::get_dom('configuration',['defaults','quotas', + &get_dom('configuration',['defaults','quotas', 'requestcourses','inststatus', 'coursedefaults','usersessions', 'requestauthor','selfenrollment', 'coursecategories','ssl','autoenroll', - 'trust','helpsettings','wafproxy','ltisec'],$domain); + 'trust','helpsettings','wafproxy', + 'ltisec','toolsec'],$domain); my @coursetypes = ('official','unofficial','community','textbook','placement'); if (ref($domconfig{'defaults'}) eq 'HASH') { $domdefaults{'lang_def'} = $domconfig{'defaults'}{'lang_def'}; @@ -2884,7 +2885,18 @@ sub get_domain_defaults { } if (ref($domconfig{'ltisec'}{'private'}) eq 'HASH') { if (ref($domconfig{'ltisec'}{'private'}{'keys'}) eq 'ARRAY') { - $domdefaults{'privhosts'} = $domconfig{'ltisec'}{'private'}{'keys'}; + $domdefaults{'ltiprivhosts'} = $domconfig{'ltisec'}{'private'}{'keys'}; + } + } + } + if (ref($domconfig{'toolsec'}) eq 'HASH') { + if (ref($domconfig{'toolsec'}{'encrypt'}) eq 'HASH') { + $domdefaults{'toolenc_crs'} = $domconfig{'toolsec'}{'encrypt'}{'crs'}; + $domdefaults{'toolenc_dom'} = $domconfig{'toolsec'}{'encrypt'}{'dom'}; + } + if (ref($domconfig{'toolsec'}{'private'}) eq 'HASH') { + if (ref($domconfig{'toolsec'}{'private'}{'keys'}) eq 'ARRAY') { + $domdefaults{'toolprivhosts'} = $domconfig{'toolsec'}{'private'}{'keys'}; } } } @@ -2907,7 +2919,7 @@ sub get_dom_cats { } else { $cats = {}; } - &Apache::lonnet::do_cache_new('cats',$dom,$cats,3600); + &do_cache_new('cats',$dom,$cats,3600); } return $cats; } @@ -2924,6 +2936,7 @@ sub get_dom_instcats { if (&auto_instcode_format($caller,$dom,\%coursecodes,\%codes, \@codetitles,\%cat_titles,\%cat_order) eq 'ok') { $instcats = { + totcodes => $totcodes, codes => \%codes, codetitles => \@codetitles, cat_titles => \%cat_titles, @@ -2962,7 +2975,7 @@ sub course_portal_url { if ($domdefaults{'portal_def'}) { $firsturl = $domdefaults{'portal_def'}; } else { - my $alias = &Apache::lonnet::use_proxy_alias($r,$chome); + my $alias = &use_proxy_alias($r,$chome); $hostname = $alias if ($alias ne ''); $firsturl = $protocol.'://'.$hostname; } @@ -3160,7 +3173,7 @@ sub courseid_to_courseurl { return "/$cdom/$cnum"; } - my %courseinfo=&Apache::lonnet::coursedescription($courseid); + my %courseinfo=&coursedescription($courseid); if (exists($courseinfo{'num'})) { return "/$courseinfo{'domain'}/$courseinfo{'num'}"; } @@ -3358,14 +3371,14 @@ sub userenvironment { # ---------------------------------------------------------- Get a studentphoto sub studentphoto { my ($udom,$unam,$ext) = @_; - my $home=&Apache::lonnet::homeserver($unam,$udom); + my $home=&homeserver($unam,$udom); if (defined($env{'request.course.id'})) { if ($env{'course.'.$env{'request.course.id'}.'.internal.showphoto'}) { if ($udom eq $env{'course.'.$env{'request.course.id'}.'.domain'}) { return(&retrievestudentphoto($udom,$unam,$ext)); } else { my ($result,$perm_reqd)= - &Apache::lonnet::auto_photo_permission($unam,$udom); + &auto_photo_permission($unam,$udom); if ($result eq 'ok') { if (!($perm_reqd eq 'yes')) { return(&retrievestudentphoto($udom,$unam,$ext)); @@ -3375,7 +3388,7 @@ sub studentphoto { } } else { my ($result,$perm_reqd) = - &Apache::lonnet::auto_photo_permission($unam,$udom); + &auto_photo_permission($unam,$udom); if ($result eq 'ok') { if (!($perm_reqd eq 'yes')) { return(&retrievestudentphoto($udom,$unam,$ext)); @@ -3387,14 +3400,14 @@ sub studentphoto { sub retrievestudentphoto { my ($udom,$unam,$ext,$type) = @_; - my $home=&Apache::lonnet::homeserver($unam,$udom); - my $ret=&Apache::lonnet::reply("studentphoto:$udom:$unam:$ext:$type",$home); + my $home=&homeserver($unam,$udom); + my $ret=&reply("studentphoto:$udom:$unam:$ext:$type",$home); if ($ret eq 'ok') { my $url="/uploaded/$udom/$unam/internal/studentphoto.$ext"; if ($type eq 'thumbnail') { $url="/uploaded/$udom/$unam/internal/studentphoto_tn.$ext"; } - my $tokenurl=&Apache::lonnet::tokenwrapper($url); + my $tokenurl=&tokenwrapper($url); return $tokenurl; } else { if ($type eq 'thumbnail') { @@ -3671,8 +3684,8 @@ sub ssi { ($form{'grade_courseid'} eq $env{'request.course.id'}) && ($form{'grade_username'} ne '') && ($form{'grade_domain'} ne '') && ($form{'grade_symb'} ne '') && - (&Apache::lonnet::allowed('mgr',$env{'request.course.id'}. - ($env{'request.course.sec'}?'/'.$env{'request.course.sec'}:'')))) { + (&allowed('mgr',$env{'request.course.id'}. + ($env{'request.course.sec'}?'/'.$env{'request.course.sec'}:'')))) { $islocal = 1; } $response= &LONCAPA::LWPReq::makerequest($lonhost,$request,'',\%perlvar, @@ -3823,7 +3836,7 @@ sub can_edit_resource { } if ($env{'request.course.id'}) { - my $crsedit = &Apache::lonnet::allowed('mdc',$env{'request.course.id'}); + my $crsedit = &allowed('mdc',$env{'request.course.id'}); if ($group ne '') { # if this is a group homepage or group bulletin board, check group privs my $allowed = 0; @@ -3852,7 +3865,7 @@ sub can_edit_resource { } } else { if ($resurl =~ m{^/?adm/viewclasslist$}) { - unless (&Apache::lonnet::allowed('opa',$env{'request.course.id'})) { + unless (&allowed('opa',$env{'request.course.id'})) { return; } } elsif (!$crsedit) { @@ -4268,7 +4281,7 @@ sub resizeImage { # input: $formname - the contents of the file are in $env{"form.$formname"} # the desired filename is in $env{"form.$formname.filename"} # $context - possible values: coursedoc, existingfile, overwrite, -# canceloverwrite, scantron or ''. +# canceloverwrite, scantron, toollogo or ''. # if 'coursedoc': upload to the current course # if 'existingfile': write file to tmp/overwrites directory # if 'canceloverwrite': delete file written to tmp/overwrites directory @@ -4280,8 +4293,8 @@ sub resizeImage { # Section => 4, CODE => 5, FirstQuestion => 9 }). # $allfiles - reference to hash for embedded objects # $codebase - reference to hash for codebase of java objects -# $desuname - username for permanent storage of uploaded file -# $dsetudom - domain for permanaent storage of uploaded file +# $destuname - username for permanent storage of uploaded file +# $destudom - domain for permanaent storage of uploaded file # $thumbwidth - width (pixels) of thumbnail to make for uploaded image # $thumbheight - height (pixels) of thumbnail to make for uploaded image # $resizewidth - width (pixels) to which to resize uploaded image @@ -4491,11 +4504,24 @@ sub finishuserfileupload { if (($thumbwidth =~ /^\d+$/) && ($thumbheight =~ /^\d+$/)) { my $input = $filepath.'/'.$file; my $output = $filepath.'/'.'tn-'.$file; + my $makethumb; my $thumbsize = $thumbwidth.'x'.$thumbheight; - my @args = ('convert','-sample',$thumbsize,$input,$output); - system({$args[0]} @args); - if (-e $filepath.'/'.'tn-'.$file) { - $fetchthumb = 1; + if ($context eq 'toollogo') { + my ($fullwidth,$fullheight) = &check_dimensions($input); + if ($fullwidth ne '' && $fullheight ne '') { + if ($fullwidth > $thumbwidth && $fullheight > $thumbheight) { + $makethumb = 1; + } + } + } else { + $makethumb = 1; + } + if ($makethumb) { + my @args = ('convert','-sample',$thumbsize,$input,$output); + system({$args[0]} @args); + if (-e $filepath.'/'.'tn-'.$file) { + $fetchthumb = 1; + } } } @@ -4727,6 +4753,30 @@ sub embedded_dependency { return; } +sub check_dimensions { + my ($inputfile) = @_; + my ($fullwidth,$fullheight); + if (($inputfile =~ m|^[/\w.\-]+$|) && (-e $inputfile)) { + my $mm = new File::MMagic; + my $mime_type = $mm->checktype_filename($inputfile); + if ($mime_type =~ m{^image/}) { + if (open(PIPE,"identify $inputfile 2>&1 |")) { + my $imageinfo = ; + if (!close(PIPE)) { + &Apache::lonnet::logthis("Failed to close PIPE opened to retrieve image information for $inputfile"); + } + chomp($imageinfo); + my ($fullsize) = + ($imageinfo =~ /^\Q$inputfile\E\s+\w+\s+(\d+x\d+)/); + if ($fullsize) { + ($fullwidth,$fullheight) = split(/x/,$fullsize); + } + } + } + } + return ($fullwidth,$fullheight); +} + sub bubblesheet_converter { my ($cdom,$fullpath,$config,$format) = @_; if ((&domain($cdom) ne '') && @@ -5134,7 +5184,7 @@ sub flushcourselogs { foreach my $entry (keys(%userrolehash)) { my ($role,$uname,$udom,$runame,$rudom,$rsec)= split(/\:/,$entry); - if (&Apache::lonnet::put('nohist_userroles', + if (&put('nohist_userroles', { $role.':'.$uname.':'.$udom.':'.$rsec => $userrolehash{$entry} }, $rudom,$runame) eq 'ok') { delete $userrolehash{$entry}; @@ -5333,7 +5383,7 @@ sub domainrolelog { my ($trole,$username,$domain,$area,$tstart,$tend,$delflag,$context)=@_; if ($area =~ m{^/($match_domain)/$}) { my $cdom = $1; - my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + my $domconfiguser = &get_domainconfiguser($cdom); my $namespace = 'rolelog'; my %storehash = ( role => $trole, @@ -5564,8 +5614,8 @@ sub get_my_adhocroles { } elsif ($cid =~ /^($match_domain)_($match_courseid)$/) { $cdom = $1; $cnum = $2; - %info = &Apache::lonnet::get('environment',['internal.coursecode'], - $cdom,$cnum); + %info = &get('environment',['internal.coursecode'], + $cdom,$cnum); } if (($info{'internal.coursecode'} ne '') && ($checkreg)) { my $user = $env{'user.name'}.':'.$env{'user.domain'}; @@ -5892,7 +5942,7 @@ sub extract_lastaccess { sub dcmailput { my ($domain,$msgid,$message,$server)=@_; - my $status = &Apache::lonnet::critical( + my $status = &critical( 'dcmailput:'.$domain.':'.&escape($msgid).'='. &escape($message),$server); return $status; @@ -7485,7 +7535,7 @@ sub putstore { '&host='.&escape($perlvar{'lonHostID'}). '&version='.$esc_v. '&by='.&escape($env{'user.name'}.':'.$env{'user.domain'}); - &Apache::lonnet::courselog($symb.':'.$uname.':'.$udomain.':PUTSTORE:'.$namevalue); + &courselog($symb.':'.$uname.':'.$udomain.':PUTSTORE:'.$namevalue); } if ($reply eq 'unknown_cmd') { # gfall back to way things use to be done @@ -7635,16 +7685,16 @@ sub get_timebased_id { my $tries = 0; # attempt to get lock on nohist_$namespace file - my $gotlock = &Apache::lonnet::newput('nohist_'.$namespace,$lockhash,$cdom,$cnum); + my $gotlock = &newput('nohist_'.$namespace,$lockhash,$cdom,$cnum); while (($gotlock ne 'ok') && $tries <$locktries) { $tries ++; sleep 1; - $gotlock = &Apache::lonnet::newput('nohist_'.$namespace,$lockhash,$cdom,$cnum); + $gotlock = &newput('nohist_'.$namespace,$lockhash,$cdom,$cnum); } # attempt to get unique identifier, based on current timestamp if ($gotlock eq 'ok') { - my %inuse = &Apache::lonnet::dump('nohist_'.$namespace,$cdom,$cnum,$prefix); + my %inuse = &dump('nohist_'.$namespace,$cdom,$cnum,$prefix); my $id = time; $newid = $id; if ($idtype eq 'addcode') { @@ -7665,7 +7715,7 @@ sub get_timebased_id { my %new_item = ( $prefix."\0".$newid => $who, ); - my $putresult = &Apache::lonnet::put('nohist_'.$namespace,\%new_item, + my $putresult = &put('nohist_'.$namespace,\%new_item, $cdom,$cnum); if ($putresult ne 'ok') { undef($newid); @@ -7954,6 +8004,17 @@ sub is_portfolio_file { return; } +sub is_coursetool_logo { + my ($uri) = @_; + if ($env{'request.course.id'}) { + my $courseurl = &courseid_to_courseurl($env{'request.course.id'}); + if ($uri =~ m{^/*uploaded\Q$courseurl\E/toollogo/\d+/[^/]+$}) { + return 1; + } + } + return; +} + sub usertools_access { my ($uname,$udom,$tool,$action,$context,$userenvref,$domdefref,$is_advref)=@_; my ($access,%tools); @@ -8104,7 +8165,7 @@ sub is_course_owner { if ($env{'course.'.$cdom.'_'.$cnum.'.internal.courseowner'} eq $uname.':'.$udom) { return 1; } else { - my %courseinfo = &Apache::lonnet::coursedescription($cdom.'/'.$cnum); + my %courseinfo = &coursedescription($cdom.'/'.$cnum); if ($courseinfo{'internal.courseowner'} eq $uname.':'.$udom) { return 1; } @@ -8582,6 +8643,12 @@ sub allowed { if ($env{'request.course.id'}) { + if ($priv eq 'bre') { + if (&is_coursetool_logo($uri)) { + return 'F'; + } + } + # If this is modifying password (internal auth) domains must match for user and user's role. if ($priv eq 'mip') { @@ -8902,13 +8969,8 @@ sub constructaccess { if (exists($env{'user.priv.au./'.$ownerdomain.'/./'})) { return ($ownername,$ownerdomain,$ownerhome); } - } else { -# Co-author for this? - if (exists($env{'user.priv.ca./'.$ownerdomain.'/'.$ownername.'./'}) || - exists($env{'user.priv.aa./'.$ownerdomain.'/'.$ownername.'./'}) ) { - $ownerhome = &homeserver($ownername,$ownerdomain); - return ($ownername,$ownerdomain,$ownerhome); - } + } elsif (&is_course($ownerdomain,$ownername)) { +# Course Authoring Space? if ($env{'request.course.id'}) { if (($ownername eq $env{'course.'.$env{'request.course.id'}.'.num'}) && ($ownerdomain eq $env{'course.'.$env{'request.course.id'}.'.domain'})) { @@ -8918,6 +8980,14 @@ sub constructaccess { } } } + return ''; + } else { +# Co-author for this? + if (exists($env{'user.priv.ca./'.$ownerdomain.'/'.$ownername.'./'}) || + exists($env{'user.priv.aa./'.$ownerdomain.'/'.$ownername.'./'}) ) { + $ownerhome = &homeserver($ownername,$ownerdomain); + return ($ownername,$ownerdomain,$ownerhome); + } } # We don't have any access right now. If we are not possibly going to do anything about this, @@ -8996,7 +9066,7 @@ sub get_comm_blocks { if ((defined($cached)) && (ref($blocksref) eq 'HASH')) { %commblocks = %{$blocksref}; } else { - %commblocks = &Apache::lonnet::dump('comm_block',$cdom,$cnum); + %commblocks = &dump('comm_block',$cdom,$cnum); my $cachetime = 600; &do_cache_new('comm_block',$hashid,\%commblocks,$cachetime); } @@ -10333,7 +10403,7 @@ sub assignrole { if ($role =~ /^cr\//) { my $cwosec=$url; $cwosec=~s/^\/($match_domain)\/($match_courseid)\/.*/$1\/$2/; - unless (&allowed('ccr',$cwosec)) { + if ((!&allowed('ccr',$cwosec)) && (!&allowed('ccr',$udom))) { my $refused = 1; if ($context eq 'requestcourses') { if (($env{'user.name'} ne '') && ($env{'user.domain'} ne '')) { @@ -10622,10 +10692,10 @@ sub store_coowners { } if (($putresult eq 'ok') || ($delresult eq 'ok')) { my %crsinfo = - &Apache::lonnet::courseiddump($cdom,'.',1,'.','.',$cnum,undef,undef,'.'); + &courseiddump($cdom,'.',1,'.','.',$cnum,undef,undef,'.'); if (ref($crsinfo{$cid}) eq 'HASH') { $crsinfo{$cid}{'co-owners'} = \@newcoowners; - my $cidput = &Apache::lonnet::courseidput($cdom,\%crsinfo,$chome,'notime'); + my $cidput = &courseidput($cdom,\%crsinfo,$chome,'notime'); } } } @@ -10843,7 +10913,7 @@ sub modifyuser { return 'error: '.$reply; } if ($names{'permanentemail'} ne $oldnames{'permanentemail'}) { - &Apache::lonnet::devalidate_cache_new('emailscache',$uname.':'.$udom); + &devalidate_cache_new('emailscache',$uname.':'.$udom); } my $sqlresult = &update_allusers_table($uname,$udom,\%names); &devalidate_cache_new('namescache',$uname.':'.$udom); @@ -10923,7 +10993,7 @@ sub modify_student_enrollment { } my $fullname = &format_name($first,$middle,$last,$gene,'lastname'); my $user = "$uname:$udom"; - my %old_entry = &Apache::lonnet::get('classlist',[$user],$cdom,$cnum); + my %old_entry = &get('classlist',[$user],$cdom,$cnum); my $reply=cput('classlist', {$user => join(':',$end,$start,$uid,$usec,$fullname,$type,$locktype,$credits,$instsec) }, @@ -11057,7 +11127,7 @@ sub createcourse { } } my %host_servers = - &Apache::lonnet::get_servers($udom,'library'); + &get_servers($udom,'library'); unless ($host_servers{$course_server}) { return 'error: invalid home server for course: '.$course_server; } @@ -12025,14 +12095,20 @@ sub stat_file { # or corresponding Published Resource Space, and populate the hash ref: # $dirhashref with URLs of all directories, and if $filehashref hash # ref arg is provided, the URLs of any files, excluding versioned, .meta, -# or .rights files in resource space, and .meta, .save, .log, and .bak -# files in Authoring Space. +# or .rights files in resource space, and .meta, .save, .log, .bak and +# .rights files in Authoring Space. # # Inputs: # # $is_home - true if current server is home server for user's space -# $context - either: priv, or res respectively for Authoring or Resource Space. -# $docroot - Document root (i.e., /home/httpd/html +# $recurse - if true will also traverse subdirectories recursively +# $include - reference to hash containing allowed file extensions. If provided, +# files which do not have a matching extension will be ignored. +# $exclude - reference to hash containing excluded file extensions. If provided, +# files which have a matching extension will be ignored. +# $nonemptydir - if true, will only populate $fileshashref hash entry for a particular +# directory with first file found (with acceptable extension). +# $addtopdir - if true, set $dirhashref->{'/'} = 1 # $toppath - Top level directory (i.e., /res/$dom/$uname or /priv/$dom/$uname # $relpath - Current path (relative to top level). # $dirhashref - reference to hash to populate with URLs of directories (Required) @@ -12049,39 +12125,61 @@ sub stat_file { # sub recursedirs { - my ($is_home,$context,$docroot,$toppath,$relpath,$dirhashref,$filehashref) = @_; + my ($is_home,$recurse,$include,$exclude,$nonemptydir,$addtopdir,$toppath,$relpath,$dirhashref,$filehashref) = @_; return unless (ref($dirhashref) eq 'HASH'); + my $docroot = $perlvar{'lonDocRoot'}; my $currpath = $docroot.$toppath; - if ($relpath) { + if ($relpath ne '') { $currpath .= "/$relpath"; } - my $savefile; + my ($savefile,$checkinc,$checkexc); if (ref($filehashref)) { $savefile = 1; } + if (ref($include) eq 'HASH') { + $checkinc = 1; + } + if (ref($exclude) eq 'HASH') { + $checkexc = 1; + } if ($is_home) { - if (opendir(my $dirh,$currpath)) { + if ((-e $currpath) && (opendir(my $dirh,$currpath))) { + my $filecount = 0; foreach my $item (sort { lc($a) cmp lc($b) } grep(!/^\.+$/,readdir($dirh))) { next if ($item eq ''); if (-d "$currpath/$item") { my $newpath; - if ($relpath) { + if ($relpath ne '') { $newpath = "$relpath/$item"; } else { $newpath = $item; } $dirhashref->{&Apache::lonlocal::js_escape($newpath)} = 1; - &recursedirs($is_home,$context,$docroot,$toppath,$newpath,$dirhashref,$filehashref); - } elsif ($savefile) { - if ($context eq 'priv') { - unless ($item =~ /\.(meta|save|log|bak|DS_Store)$/) { - $filehashref->{&Apache::lonlocal::js_escape($relpath)}{$item} = 1; + if ($recurse) { + &recursedirs($is_home,$recurse,$include,$exclude,$nonemptydir,$addtopdir,$toppath,$newpath,$dirhashref,$filehashref); + } + } elsif (($savefile) || ($relpath eq '')) { + next if ($nonemptydir && $filecount); + if ($checkinc || $checkexc) { + my ($extension) = ($item =~ /\.(\w+)$/); + if ($checkinc) { + next unless ($extension && $include->{$extension}); } - } else { - unless (($item =~ /\.meta$/) || ($item =~ /\.\d+\.\w+$/) || ($item =~ /\.rights$/)) { + if ($checkexc) { + next if ($extension && $exclude->{$extension}); + } + } + if (($relpath eq '') && (!exists($dirhashref->{'/'}))) { + $dirhashref->{'/'} = 1; + } + if ($savefile) { + if ($relpath eq '') { + $filehashref->{'/'}{$item} = 1; + } else { $filehashref->{&Apache::lonlocal::js_escape($relpath)}{$item} = 1; } } + $filecount ++; } } closedir($dirh); @@ -12092,6 +12190,7 @@ sub recursedirs { my @dir_lines; my $dirptr=16384; if (ref($dirlistref) eq 'ARRAY') { + my $filecount = 0; foreach my $dir_line (sort { my ($afile)=split('&',$a,2); @@ -12107,28 +12206,57 @@ sub recursedirs { if ($relpath) { $newpath = "$relpath/$item"; } else { - $relpath = '/'; $newpath = $item; } $dirhashref->{&Apache::lonlocal::js_escape($newpath)} = 1; - &recursedirs($is_home,$context,$docroot,$toppath,$newpath,$dirhashref,$filehashref); - } elsif ($savefile) { - if ($context eq 'priv') { - unless ($item =~ /\.(meta|save|log|bak|DS_Store)$/) { - $filehashref->{$relpath}{$item} = 1; + if ($recurse) { + &recursedirs($is_home,$recurse,$include,$exclude,$nonemptydir,$addtopdir,$toppath,$newpath,$dirhashref,$filehashref); + } + } elsif (($savefile) || ($relpath eq '')) { + next if ($nonemptydir && $filecount); + if ($checkinc || $checkexc) { + my $extension; + if ($checkinc) { + next unless ($extension && $include->{$extension}); } - } else { - unless (($item =~ /\.meta$/) || ($item =~ /\.\d+\.\w+$/)) { - $filehashref->{$relpath}{$item} = 1; + if ($checkexc) { + next if ($extension && $exclude->{$extension}); } } + if (($relpath eq '') && (!exists($dirhashref->{'/'}))) { + $dirhashref->{'/'} = 1; + } + if ($savefile) { + if ($relpath eq '') { + $filehashref->{'/'}{$item} = 1; + } else { + $filehashref->{&Apache::lonlocal::js_escape($relpath)}{$item} = 1; + } + } + $filecount ++; } } } } + if ($addtopdir) { + if (($relpath eq '') && (!exists($dirhashref->{'/'}))) { + $dirhashref->{'/'} = 1; + } + } return; } +sub priv_exclude { + return { + meta => 1, + save => 1, + log => 1, + bak => 1, + rights => 1, + DS_Store => 1, + }; +} + # -------------------------------------------------------- Value of a Condition # gets the value of a specific preevaluated condition @@ -12457,25 +12585,64 @@ sub domainlti_itemid { return $itemid; } -sub get_numsuppfiles { - my ($cnum,$cdom,$ignorecache)=@_; +sub count_supptools { + my ($cnum,$cdom,$ignorecache,$reload)=@_; my $hashid=$cnum.':'.$cdom; - my ($suppcount,$cached); + my ($numexttools,$cached); unless ($ignorecache) { - ($suppcount,$cached) = &is_cached_new('suppcount',$hashid); + ($numexttools,$cached) = &is_cached_new('supptools',$hashid); } unless (defined($cached)) { my $chome=&homeserver($cnum,$cdom); + $numexttools = 0; unless ($chome eq 'no_host') { - ($suppcount,my $supptools,my $errors) = (0,0,0); - my $suppmap = 'supplemental.sequence'; - ($suppcount,$supptools,$errors) = - &Apache::loncommon::recurse_supplemental($cnum,$cdom,$suppmap,$suppcount, - $supptools,$errors); + my ($supplemental) = &Apache::loncommon::get_supplemental($cnum,$cdom,$reload); + if (ref($supplemental) eq 'HASH') { + if ((ref($supplemental->{'ids'}) eq 'HASH') && (ref($supplemental->{'hidden'}) eq 'HASH')) { + foreach my $key (keys(%{$supplemental->{'ids'}})) { + if ($key =~ m{^/adm/$cdom/$cnum/\d+/ext\.tool$}) { + $numexttools ++; + } + } + } + } } - &do_cache_new('suppcount',$hashid,$suppcount,600); + &do_cache_new('supptools',$hashid,$numexttools,600); } - return $suppcount; + return $numexttools; +} + +sub has_unhidden_suppfiles { + my ($cnum,$cdom,$ignorecache,$possdel)=@_; + my $hashid=$cnum.':'.$cdom; + my ($showsupp,$cached); + unless ($ignorecache) { + ($showsupp,$cached) = &is_cached_new('showsupp',$hashid); + } + unless (defined($cached)) { + my $chome=&homeserver($cnum,$cdom); + unless ($chome eq 'no_host') { + my ($supplemental) = &Apache::loncommon::get_supplemental($cnum,$cdom,$ignorecache,$possdel); + if (ref($supplemental) eq 'HASH') { + if ((ref($supplemental->{'ids'}) eq 'HASH') && (ref($supplemental->{'hidden'}) eq 'HASH')) { + foreach my $key (keys(%{$supplemental->{'ids'}})) { + next if ($key =~ /\.sequence$/); + if (ref($supplemental->{'ids'}->{$key}) eq 'ARRAY') { + foreach my $id (@{$supplemental->{'ids'}->{$key}}) { + unless ($supplemental->{'hidden'}->{$id}) { + $showsupp = 1; + last; + } + } + } + last if ($showsupp); + } + } + } + } + &do_cache_new('showsupp',$hashid,$showsupp,600); + } + return $showsupp; } # @@ -13551,13 +13718,13 @@ sub get_reservable_slots { sub get_course_slots { my ($cnum,$cdom) = @_; my $hashid=$cnum.':'.$cdom; - my ($result,$cached) = &Apache::lonnet::is_cached_new('allslots',$hashid); + my ($result,$cached) = &is_cached_new('allslots',$hashid); if (defined($cached)) { if (ref($result) eq 'HASH') { return %{$result}; } } else { - my %slots=&Apache::lonnet::dump('slots',$cdom,$cnum); + my %slots=&dump('slots',$cdom,$cnum); my ($tmp) = keys(%slots); if ($tmp !~ /^(con_lost|error|no_such_host)/i) { &do_cache_new('allslots',$hashid,\%slots,600); @@ -13603,11 +13770,68 @@ sub get_coursechange { } sub devalidate_coursechange_cache { - my ($cnum,$cdom)=@_; - my $hashid=$cnum.':'.$cdom; + my ($cdom,$cnum)=@_; + my $hashid=$cdom.'_'.$cnum; &devalidate_cache_new('crschange',$hashid); } +sub get_suppchange { + my ($cdom,$cnum) = @_; + if ($cdom eq '' || $cnum eq '') { + return unless ($env{'request.course.id'}); + $cnum = $env{'course.'.$env{'request.course.id'}.'.num'}; + $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'}; + } + my $hashid=$cdom.'_'.$cnum; + my ($change,$cached)=&is_cached_new('suppchange',$hashid); + if ((defined($cached)) && ($change ne '')) { + return $change; + } else { + my %crshash = &get('environment',['internal.supplementalchange'],$cdom,$cnum); + if ($crshash{'internal.supplementalchange'} eq '') { + $change = $env{'course.'.$cdom.'_'.$cnum.'.internal.created'}; + if ($change eq '') { + %crshash = &get('environment',['internal.created'],$cdom,$cnum); + $change = $crshash{'internal.created'}; + } + } else { + $change = $crshash{'internal.supplementalchange'}; + } + my $cachetime = 600; + &do_cache_new('suppchange',$hashid,$change,$cachetime); + } + return $change; +} + +sub devalidate_suppchange_cache { + my ($cdom,$cnum)=@_; + my $hashid=$cdom.'_'.$cnum; + &devalidate_cache_new('suppchange',$hashid); +} + +sub update_supp_caches { + my ($cdom,$cnum) = @_; + my %servers = &internet_dom_servers($cdom); + my @ids=¤t_machine_ids(); + foreach my $server (keys(%servers)) { + next if (grep(/^\Q$server\E$/,@ids)); + my $hashid=$cnum.':'.$cdom; + my $cachekey = &escape('showsupp').':'.&escape($hashid); + &remote_devalidate_cache($server,[$cachekey]); + } + &has_unhidden_suppfiles($cnum,$cdom,1,1); + &count_supptools($cnum,$cdom,1); + my $now = time; + if ($env{'request.course.id'} eq $cdom.'_'.$cnum) { + &Apache::lonnet::appenv({'request.course.suppupdated' => $now}); + } + &put('environment',{'internal.supplementalchange' => $now}, + $cdom,$cnum); + &Apache::lonnet::appenv( + {'course.'.$cdom.'_'.$cnum.'.internal.supplementalchange' => $now}); + &do_cache_new('suppchange',$cdom.'_'.$cnum,$now,600); +} + # ------------------------------------------------- Update symbolic store links sub symblist { @@ -14771,7 +14995,7 @@ sub get_requestor_ip { sub get_proxy_settings { my ($dom_in_use) = @_; - my %domdefaults = &Apache::lonnet::get_domain_defaults($dom_in_use); + my %domdefaults = &get_domain_defaults($dom_in_use); my $proxyinfo = { ipheader => $domdefaults{'waf_ipheader'}, trusted => $domdefaults{'waf_trusted'}, @@ -14804,11 +15028,11 @@ sub get_proxy_alias { if ($cached) { return $alias; } - my $dom = &Apache::lonnet::host_domain($lonid); + my $dom = &host_domain($lonid); if ($dom ne '') { my $cachetime = 60*60*24; my %domconfig = - &Apache::lonnet::get_dom('configuration',['wafproxy'],$dom); + &get_dom('configuration',['wafproxy'],$dom); if (ref($domconfig{'wafproxy'}) eq 'HASH') { if (ref($domconfig{'wafproxy'}{'alias'}) eq 'HASH') { $alias = $domconfig{'wafproxy'}{'alias'}{$lonid}; @@ -14855,11 +15079,11 @@ sub alias_sso { if ($cached) { return $use_alias; } - my $dom = &Apache::lonnet::host_domain($lonid); + my $dom = &host_domain($lonid); if ($dom ne '') { my $cachetime = 60*60*24; my %domconfig = - &Apache::lonnet::get_dom('configuration',['wafproxy'],$dom); + &get_dom('configuration',['wafproxy'],$dom); if (ref($domconfig{'wafproxy'}) eq 'HASH') { if (ref($domconfig{'wafproxy'}{'saml'}) eq 'HASH') { $use_alias = $domconfig{'wafproxy'}{'saml'}{$lonid}; @@ -14887,7 +15111,7 @@ sub get_saml_landing { $lonid = $perlvar{'lonHostID'}; } if ($lonid) { - unless (&Apache::lonnet::host_domain($lonid) eq $defdom) { + unless (&host_domain($lonid) eq $defdom) { return; } } else { @@ -14900,11 +15124,11 @@ sub get_saml_landing { if ($cached) { return $landing; } - my $dom = &Apache::lonnet::host_domain($lonid); + my $dom = &host_domain($lonid); if ($dom ne '') { my $cachetime = 60*60*24; my %domconfig = - &Apache::lonnet::get_dom('configuration',['login'],$dom); + &get_dom('configuration',['login'],$dom); if (ref($domconfig{'login'}) eq 'HASH') { if (ref($domconfig{'login'}{'saml'}) eq 'HASH') { if (ref($domconfig{'login'}{'saml'}{$lonid}) eq 'HASH') { @@ -15034,7 +15258,7 @@ sub get_dns { my ($url,$func,$ignore_cache,$nocache,$hashref) = @_; if (!$ignore_cache) { my ($content,$cached)= - &Apache::lonnet::is_cached_new('dns',$url); + &is_cached_new('dns',$url); if ($cached) { &$func($content,$hashref); return; @@ -15113,7 +15337,7 @@ sub get_dns { sub parse_dns_checksums_tab { my ($lines,$hashref) = @_; my $lonhost = $perlvar{'lonHostID'}; - my $machine_dom = &Apache::lonnet::host_domain($lonhost); + my $machine_dom = &host_domain($lonhost); my $loncaparev = &get_server_loncaparev($machine_dom); my $distro = (split(/\:/,&get_server_distarch($lonhost)))[0]; my $webconfdir = '/etc/httpd/conf'; @@ -15157,7 +15381,7 @@ sub parse_dns_checksums_tab { sub fetch_dns_checksums { my %checksums; - my $machine_dom = &Apache::lonnet::host_domain($perlvar{'lonHostID'}); + my $machine_dom = &host_domain($perlvar{'lonHostID'}); my $loncaparev = &get_server_loncaparev($machine_dom,$perlvar{'lonHostID'}); my ($release,$timestamp) = split(/\-/,$loncaparev); &get_dns("/adm/dns/checksums/$release",\&parse_dns_checksums_tab,1,1, @@ -15545,7 +15769,7 @@ sub parse_getdns_url { return %iphost; } my ($ip_info,$cached)= - &Apache::lonnet::is_cached_new('iphost','iphost'); + &is_cached_new('iphost','iphost'); if ($cached) { %iphost = %{$ip_info->[0]}; %name_to_ip = %{$ip_info->[1]}; @@ -15557,7 +15781,7 @@ sub parse_getdns_url { # get yesterday's info for fallback my %old_name_to_ip; my ($ip_info,$cached)= - &Apache::lonnet::is_cached_new('iphost','iphost'); + &is_cached_new('iphost','iphost'); if ($cached) { %old_name_to_ip = %{$ip_info->[1]}; } @@ -15624,7 +15848,7 @@ sub parse_getdns_url { my ($lonid) = @_; return if ($lonid eq ''); my ($idnref,$cached)= - &Apache::lonnet::is_cached_new('internetnames',$lonid); + &is_cached_new('internetnames',$lonid); if ($cached) { return $idnref; } @@ -16464,10 +16688,6 @@ data base, returning a hash that is keye values that are the resource value. I believe that the timestamps and versions are also returned. -get_numsuppfiles($cnum,$cdom) : retrieve number of files in a course's -supplemental content area. This routine caches the number of files for -10 minutes. - =back =head2 Course Modification