--- loncom/interface/loncoursequeueadmin.pm 2009/10/04 16:07:11 1.9.2.2 +++ loncom/interface/loncoursequeueadmin.pm 2025/03/18 18:57:28 1.70 @@ -1,7 +1,7 @@ # The LearningOnline Network -# Utilities to administer domain course requests and course self-enroll requests +# Utilities to administer domain course requests and course self-enroll requests # -# $Id: loncoursequeueadmin.pm,v 1.9.2.2 2009/10/04 16:07:11 raeburn Exp $ +# $Id: loncoursequeueadmin.pm,v 1.70 2025/03/18 18:57:28 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -27,13 +27,17 @@ # ### +=pod + =head1 NAME Apache::loncoursequeueadmin.pm =head1 SYNOPSIS -Adminitsration utilities used by domain coordinators for queued course creation requests, and by course coordinators for queued self-enrollment requests. +Utilities used by domain coordinators to administer queued course creation requests, +and by course coordinators for queued self-enrollment requests, and by general +users to display their queued self-enrollment requests. This is part of the LearningOnline Network with CAPA project described at http://www.lon-capa.org. @@ -46,10 +50,34 @@ described at http://www.lon-capa.org. =item display_queued_requests() +=item build_queue_display() + =item update_request_queue() =item get_student_counts() +=item course_creation() + +=item build_batchcreatehash() + +=item can_clone_course() + +=item get_processtype() + +=item queued_selfenrollment() + +=item update_coursereq_status() + +=item process_official_reqs() + +=item is_active_author() + +=item author_prompt() + +=item reqauthor_check() + +=item process_reqauthor() + =back =cut @@ -57,86 +85,235 @@ described at http://www.lon-capa.org. package Apache::loncoursequeueadmin; use strict; -use Apache::Constants qw(:common :http); use Apache::lonnet; use Apache::loncommon; use Apache::lonmsg; use Apache::lonlocal; use Apache::lonuserutils; -use LONCAPA; +use LONCAPA::batchcreatecourse; +use LONCAPA qw(:DEFAULT :match); sub send_selfserve_notification { my ($notifylist,$textstr,$cid,$contextdesc,$timestamp,$context,$sender, - $approvedlist,$rejectedlist) = @_; + $approvedlist,$rejectedlist,$crstype) = @_; # FIXME locallocaltime needs to be able to take $sender_lh as an argument # so this can be localized to the recipients date display format/time zone $timestamp =&Apache::lonlocal::locallocaltime($timestamp); - my $msgcc; - my ($rawsubj,@rawmsg,$subject,$message,$reviewer); + my ($msgcc,$rawsubj,@rawmsg,$subject,$message,$reviewer,$msgtxt); + my ($senderuname,$senderudom) = split(':',$sender); if ($context eq 'coursemanagers') { $rawsubj = 'Self-enrollment requests processed'; push(@rawmsg,{ - mt => 'Enrollment requests in the following course: [_1]have been processed.', - args => ["\n $contextdesc"], + mt => 'Enrollment requests in the following course: [_1] have been processed.', + args => ["\n$contextdesc\n"], }); } elsif ($context eq 'domainmanagers') { - $rawsubj = 'Course requests reviewed'; + $rawsubj = 'Course/Community requests reviewed'; + push(@rawmsg,{ + mt => 'Course/Community creation requests in the following domain: [_1] have been reviewed.', + args => ["\n$contextdesc\n"], + }); + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'authormanagers') { + $rawsubj = 'Authoring Space requests reviewed'; + push(@rawmsg,{ + mt => 'Authoring requests in the following domain: [_1] have been reviewed.', + args => ["\n$contextdesc\n"], + }); + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'usernamemanagers') { + $rawsubj = 'LON-CAPA account requests reviewed'; push(@rawmsg,{ - mt => 'Course creation requests in the following domain: [_1]have been reviewed.', - args => ["\n $contextdesc"], + mt => 'Account requests in the following domain: [_1] have been reviewed.', + args => ["\n$contextdesc\n"], }); if (ref($textstr) eq 'ARRAY') { push(@rawmsg,@{$textstr}); } } elsif ($context eq 'enroller') { $rawsubj = 'Enrollment request'; + if ($crstype eq 'community') { + $msgtxt = 'Your request for enrollment in the following community: [_1]requested on [_2]has been reviewed by a Coordinator.' + } else { + $msgtxt = 'Your request for enrollment in the following course: [_1]requested on [_2]has been reviewed by a Course Coordinator.'; + } push(@rawmsg,{ - mt => 'Your request for enrollment in the following course: [_1]requested on [_2]has been reviewed by a Course Coordinator.', + mt => $msgtxt, args => ["\n ".$contextdesc.",\n",$timestamp.",\n"], - }); if (ref($textstr) eq 'ARRAY') { push(@rawmsg,@{$textstr}); } } elsif ($context eq 'courserequestor') { - $rawsubj = 'Course request'; + if ($crstype eq 'Community') { + $rawsubj = 'Community request'; + $msgtxt = 'Your request for creation of the following community: [_1]requested on [_2]has been reviewed by a Domain Coordinator.'; + } else { + $rawsubj = 'Course request'; + $msgtxt = 'Your request for creation of the following course: [_1]requested on [_2]has been reviewed by a Domain Coordinator.'; + } push(@rawmsg,{ - mt => 'Your request for creation of the following course: [_1]requested on [_2]has been reviewed by a Domain Coordinator.', + mt => $msgtxt, args => ["\n".$contextdesc.",\n",$timestamp.",\n"], }); if (ref($textstr) eq 'ARRAY') { push(@rawmsg,@{$textstr}); } + } elsif ($context eq 'pendingrequestor') { + if ($crstype eq 'Community') { + $rawsubj = 'Community request'; + } else { + $rawsubj = 'Processed course request'; + } + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } } elsif ($context eq 'coursereq') { - $rawsubj = 'Course request to review', + if ($crstype eq 'community') { + $rawsubj = 'Community request to review'; + $msgtxt = 'Creation of the following community: [_1]was requested by [_2] on [_3].'; + } else { + $rawsubj = 'Course request to review'; + $msgtxt = 'Creation of the following course: [_1]was requested by [_2] on [_3].'; + } push(@rawmsg,{ - mt => 'Creation of the following course: [_1]was requested by [_2] on [_3].', + mt => $msgtxt, args => ["\n $contextdesc\n",$textstr,$timestamp], }, { - mt =>'[_1]As Domain Coordinator, use: [_2]Main Menu -> Create a new course -> Approve or reject course requests[_3]to display a list of pending requests, which you can either approve or reject.', - args => ["\n","\n\n ","\n\n"], + mt =>'[_1]As Domain Coordinator, use: [_2]Main Menu -> Course and community creation -> Approve or reject requests[_3]to display a list of pending requests, which you can either approve or reject.', + args => ["\n","\n\n","\n\n"], }); } elsif ($context eq 'selfenrollreq') { $rawsubj = 'Self-enrollment request'; + if ($crstype eq 'community') { + $msgtxt = 'Enrollment in the following community: [_1]was requested by [_2] on [_3].' + } else { + $msgtxt = 'Enrollment in the following course: [_1]was requested by [_2] on [_3].' + } push(@rawmsg,{ - mt => 'Enrollment in the following course: [_1] was requested by [_2] on [_3].', + mt => $msgtxt, args => ["\n $contextdesc\n",$textstr,$timestamp."\n"], }); - if ($env{'course.'.$env{'request.course.id'}.'.type'} eq 'Community') { - push(@rawmsg, + my $directions; + if ($crstype eq 'community') { + $directions = 'As Coordinator, use: [_1]Main Menu -> Manage Community Users -> Enrollment Requests[_2]to display a list of pending enrollment requests, which you can either approve or reject.'; + } else { + $directions = 'As Course Coordinator, use: [_1]Main Menu -> Manage Course Users -> Enrollment Requests[_2]to display a list of pending enrollment requests, which you can either approve or reject.'; + } + push(@rawmsg, { - mt =>'As Coordinator, use: [_1]Main Menu -> Manage Course Users -> Enrollment Requests[_2]to display a list of pending enrollment requests, which you can either approve or reject.', + mt => $directions, args => [" \n\n","\n"], }); - } else { - push(@rawmsg, + } elsif ($context eq 'authorreq') { + $rawsubj = 'Authoring Space request to review'; + $msgtxt = 'Assignment of an author role in the [_1] domain[_2]was requested by [_3] on [_4].'; + push(@rawmsg,{ + mt => $msgtxt, + args => [$contextdesc,"\n",$textstr,$timestamp], + }, { - mt =>'As Course Coordinator, use: [_1]Main Menu -> Manage Course Users -> Enrollment Requests[_2]to display a list of pending enrollment requests, which you can either approve or reject.', - args => [" \n\n","\n"], + mt =>'[_1]As Domain Coordinator, use: [_2]Main Menu -> Create users or modify the roles and privileges of users -> Authoring Space requests[_3]to display a list of pending requests, which you can either approve or reject.', + args => ["\n","\n\n ","\n\n"], }); - + } elsif ($context eq 'requestauthor') { + $rawsubj = 'Authoring Space request'; + $msgtxt = 'Your request for an Authoring Space requested on [_1]has been reviewed by a Domain Coordinator.'; + push(@rawmsg,{ + mt => $msgtxt, + args => [$timestamp."\n"], + }); + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'usernamereq') { + $rawsubj = 'LON-CAPA account request'; + $msgtxt = 'Creation of a LON-CAPA account in the [_1] domain[_2]was requested by [_3] on [_4].'; + push(@rawmsg,{ + mt => $msgtxt, + args => [$contextdesc,"\n",$textstr,$timestamp], + }, + { + mt =>'[_1]As Domain Coordinator, use: [_2]Main Menu -> Create users or modify the roles and privileges of users + -> LON-CAPA account requests[_3]to display a list of pending requests, which you can either approve or reject.', + args => ["\n","\n\n ","\n\n"], + }); + } elsif ($context eq 'requestusername') { + $rawsubj = 'LON-CAPA account request'; + $msgtxt = 'Your request for a LON-CAPA account requested on [_1]has been reviewed by a Domain Coordinator.'; + push(@rawmsg,{ + mt => $msgtxt, + args => [$timestamp."\n"], + }); + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'uniquecode') { + $rawsubj = 'Course Identifier'; + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'queuedcrsreq') { + $rawsubj = 'Course Request Queued'; + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'createdcrsreq') { + $rawsubj = 'Course Creation Information'; + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); + } + } elsif ($context eq 'othdomroleuser') { + my $linktext = 'Roles'; + if ($notifylist =~ /^($match_username):($match_domain)$/) { + if (&Apache::loncommon::show_course($2,$1)) { + $linktext = 'Courses'; + } + } + $rawsubj = 'Role Assignment Approval'; + push(@rawmsg,{ + mt => 'A domain different to your own LON-CAPA domain [_1]wants to assign you a role in their domain.', + args => ["\n$contextdesc\n"], + }, + { + mt =>"[_1]Click $linktext at top right, then click 'Show pending' in the gray Functions bar ". + "to display a list of pending role assignments in other domain(s), which you can either accept or reject.", + args => ["\n\n"], + }); + } elsif ($context eq 'othdomroledc') { + $rawsubj = 'Role Assignment Authorization'; + push(@rawmsg,{ + mt => 'Another LON-CAPA domain wants to assign a role in their domain to a user from your domain.', + args => [], + }, + { + mt =>'[_1]As Domain Coordinator, use: [_2]Main Menu -> Create users or modify the roles and privileges of users + -> Queued Role Assignments (this domain) [_3]to display a list of pending requests, which you can either approve or reject.', + args => ["\n","\n\n ","\n\n"], + }); + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + $rawsubj = 'Status of Role Assignment Requests'; + if ($context eq 'othdombydc') { + push(@rawmsg,{ + mt =>'A Domain Coordinator in a domain different to your own LON-CAPA domain '. + 'has taken action on queued role assignment(s) in this domain for user(s) from that other domain [_1]', + args => ["\n$contextdesc\n"], + }); + } elsif ($context eq 'othdombyuser') { + push(@rawmsg,{ + mt => "Action has been taken by a user to whom you had assigned role(s) ". + "which had been queued, pending acceptance of the role(s).", + args => [], + }); + } + if (ref($textstr) eq 'ARRAY') { + push(@rawmsg,@{$textstr}); } } my @to_notify = split(/,/,$notifylist); @@ -155,9 +332,14 @@ sub send_selfserve_notification { my %reciphash = ( cc => $msgcc, ); - my ($uname,$udom); + my ($uname,$udom,$need_temp_env); if ($sender =~ /:/) { ($uname,$udom) = split(/:/,$sender); + if ($context eq 'usernamereq') { + unless ($env{'user.name'} && $env{'user.domain'}) { + $need_temp_env = 1; + } + } } elsif ($context eq 'course') { $uname = $sender; my %courseinfo = &Apache::lonnet::coursedescription($cid); @@ -167,198 +349,857 @@ sub send_selfserve_notification { my $stamp = time; my $msgcount = &Apache::lonmsg::get_uniq(); my $sender_lh = &Apache::loncommon::user_lang($uname,$udom,$cid); - $subject = &Apache::lonlocal::mt_user($sender_lh,$rawsubj); + $subject = &mt_user($sender_lh,$rawsubj); $message = ''; foreach my $item (@rawmsg) { if (ref($item) eq 'HASH') { - $message .= &Apache::lonlocal::mt_user($sender_lh,$item->{mt},@{$item->{args}})."\n"; + if (ref($item->{args}) eq 'ARRAY') { + $message .= &mt_user($sender_lh,$item->{mt},@{$item->{args}})."\n"; + } else { + $message .= &mt_user($sender_lh,$item->{mt})."\n"; + } } } - &Apache::lonmsg::process_sent_mail($subject,'',$numsent,$stamp,$uname,$udom,$msgcount,$cid,$$,$message,\@recusers,\@recudoms); + &Apache::lonmsg::process_sent_mail($subject,'',$numsent,$stamp,$uname,$udom,$msgcount,$cid,$$,$message, + \@recusers,\@recudoms,undef,undef,undef,undef,$senderuname,$senderudom); my ($recipid,$recipstatus) = &Apache::lonmsg::store_recipients($subject,$uname,$udom,\%reciphash); my $status; + if ($need_temp_env) { + $env{'user.name'} = $uname; + $env{'user.domain'} = $udom; + } foreach my $recip (sort(keys(%{$msgcc}))) { my ($ccname,$ccdom) = split(/:/,$recip); my $recip_lh = &Apache::loncommon::user_lang($ccname,$ccdom,$cid); - my $subject = &Apache::lonlocal::mt_user($sender_lh,$rawsubj); + my $subject = &mt_user($sender_lh,$rawsubj); my $message = ''; foreach my $item (@rawmsg) { if (ref($item) eq 'HASH') { - $message .= &Apache::lonlocal::mt_user($sender_lh,$item->{mt}, - @{$item->{args}})."\n"; + $message .= &mt_user($sender_lh,$item->{mt}, + @{$item->{args}})."\n"; } } - if ($context eq 'managers') { + if ($context eq 'coursemanagers') { if ($approvedlist) { - $message .= "\n\n".&Apache::lonlocal::mt_user($sender_lh,'Approved enrollments:')."\n".$approvedlist; + $message .= "\n\n".&mt_user($sender_lh,'Approved enrollments:')."\n".$approvedlist; } if ($rejectedlist) { - $message .= "\n\n".&Apache::lonlocal::mt_user($sender_lh,'Rejected enrollments:')."\n".$rejectedlist; + $message .= "\n\n".&mt_user($sender_lh,'Rejected enrollments:')."\n".$rejectedlist; } } elsif ($context eq 'domainmanagers') { if ($approvedlist) { - $message .= "\n\n".&Apache::lonlocal::mt_user($sender_lh,'Approved course requests:')."\n".$approvedlist; + $message .= "\n\n".&mt_user($sender_lh,'Approved course requests:')."\n".$approvedlist; + } + if ($rejectedlist) { + $message .= "\n\n".&mt_user($sender_lh,'Rejected course requests:')."\n".$rejectedlist; + } + } elsif ($context eq 'authormanagers') { + if ($approvedlist) { + $message .= "\n\n".&mt_user($sender_lh,'Approved author role requests:')."\n".$approvedlist; } if ($rejectedlist) { - $message .= "\n\n".&Apache::lonlocal::mt_user($sender_lh,'Rejected course requests:')."\n".$rejectedlist; + $message .= "\n\n".&mt_user($sender_lh,'Rejected author role requests:')."\n".$rejectedlist; + } + } elsif ($context eq 'usernamemanagers') { + if ($approvedlist) { + $message .= "\n\n".&mt_user($sender_lh,'Approved LON-CAPA account requests:')."\n".$approvedlist; + } + if ($rejectedlist) { + $message .= "\n\n".&mt_user($sender_lh,'Rejected LON-CAPA account requests:')."\n".$rejectedlist; } } - $status .= &Apache::lonmsg::user_normal_msg($ccname,$ccdom,$subject,$message,undef,undef,undef,1,\%sentmessage,undef,undef,undef,1,$recipid).','; + $status .= &Apache::lonmsg::user_normal_msg($ccname,$ccdom,$subject,$message,undef,undef,undef,1, + \%sentmessage,undef,undef,undef,1,$recipid).','; } $status =~ s/,$//; + if ($need_temp_env) { + undef($env{'user.name'}); + undef($env{'user.domain'}); + } return ($recipstatus,$status); } sub display_queued_requests { - my ($context,$dom,$cnum) = @_; - my ($namespace,$formaction,$nextelement,%requesthash); + my ($context,$dom,$cnum,$secondary) = @_; + my ($namespace,$formaction,$nextelement,%requesthash,%reqstatus); if ($context eq 'course') { $formaction = '/adm/createuser'; $namespace = 'selfenrollrequests'; %requesthash = &Apache::lonnet::dump($namespace,$dom,$cnum); $nextelement = '<input type="hidden" name="state" value="done" />'; + } elsif ($context eq 'requestauthor') { + $formaction = '/adm/createuser'; + $namespace = 'requestauthorqueue'; + %requesthash = &Apache::lonnet::dump_dom($namespace,$dom); + $nextelement = '<input type="hidden" name="state" value="done" />'; + } elsif ($context eq 'requestusername') { + $formaction = '/adm/createuser'; + $namespace = 'usernamequeue'; + %requesthash = &Apache::lonnet::dump_dom($namespace,$dom); + $nextelement = '<input type="hidden" name="state" value="done" />'; + } elsif ($context eq 'othdomqueue') { + $formaction = '/adm/createuser'; + $namespace = 'nohist_othdomqueued'; + if ($secondary eq 'domain') { + %requesthash = &Apache::lonnet::dump_dom($namespace,$dom); + foreach my $key (keys(%requesthash)) { + delete($requesthash{$key}) if ($key =~ /:(ca|aa)$/); + } + } elsif ($secondary eq 'author') { + %requesthash = &Apache::lonnet::dump($namespace,$dom,$cnum); + if ($cnum eq &Apache::lonnet::get_domainconfiguser($dom)) { + foreach my $key (keys(%requesthash)) { + delete($requesthash{$key}) if ($key !~ /:(ca|aa)$/); + } + } + } else { + %requesthash = &Apache::lonnet::dump($namespace,$dom,$cnum); + } + } elsif ($context eq 'othdomaction') { + $namespace = 'nohist_queuedrolereqs'; + if ($secondary eq 'domain') { + $formaction = '/adm/createuser'; + my $confname = &Apache::lonnet::get_domainconfiguser($dom); + %requesthash = &Apache::lonnet::dump($namespace,$dom,$confname); + } else { + $formaction = '/adm/roles'; + %requesthash = &Apache::lonnet::dump($namespace,$dom,$cnum); + } + $nextelement = '<input type="hidden" name="state" value="done" />'; } else { $formaction = '/adm/createcourse'; $namespace = 'courserequestqueue'; - %requesthash = &Apache::lonnet::dump_dom($namespace,$dom,'_approval'); - $nextelement = '<input type="hidden" name="phase" value="requestchange" />'; + my $disposition = 'approval'; + my $nextphase = 'requestchange'; + if ($context eq 'pending') { + $disposition = 'pending'; + $nextphase = 'requestvalidation'; + } elsif ($context eq 'displaypending') { + $disposition = 'pending'; + } + %requesthash = &Apache::lonnet::dump_dom($namespace,$dom,'_'.$disposition); + $nextelement = '<input type="hidden" name="phase" value="'.$nextphase.'" />'; } - my ($output,%queue_by_date,%crstypes); + my ($output,%queue_by_date); if (keys(%requesthash) > 0) { - $output = '<form method="post" name="changequeue" action="'.$formaction.'" />'."\n". - '<input type="hidden" name="action" value="'.$env{'form.action'}.'" />'."\n". - $nextelement."\n". - &Apache::loncommon::start_data_table(). - &Apache::loncommon::start_data_table_header_row(). - '<th>'.&mt('Action').'</th>'. - '<th>'.&mt('Requestor').'</th>'; - if ($context eq 'course') { - $output .= '<th>'.&mt('Section').'</th>'. - '<th>'.&mt('Date requested').'</th>'; - %crstypes = &Apache::lonlocal::texthash ( - official => 'Official course', - unofficial => 'Unofficial course', - community => 'Community', - ); - } else { - $output .= '<th>'.&mt('Type').'</th>'. - '<th>'.&mt('Date requested').'</th>'. - '<th>'.&mt('Details').'</th>'; - } - $output .= &Apache::loncommon::end_data_table_header_row(); foreach my $item (keys(%requesthash)) { - my ($timestamp,$entry); + my ($timestamp,$entry,$pending); if ($context eq 'course') { ($timestamp, my $usec) = split(/:/,$requesthash{$item}); $entry = $item.':'.$usec; + } elsif ($context eq 'requestauthor') { + $timestamp = $requesthash{$item}; + ($entry) = ($item =~ /^($match_username)_approval$/); + } elsif ($context eq 'requestusername') { + $timestamp = $requesthash{$item}; + ($entry) = (&unescape($item) =~ /^($match_username)_approval$/); + } elsif ($context eq 'othdomqueue') { + if (ref($requesthash{$item}) eq 'HASH') { + my ($puname,$pudom,$prole,$psec) = split(/:/,$item); + $timestamp = $requesthash{$item}{'timestamp'}; + my $adj = $requesthash{$item}{'adj'}; + $entry = join(':',$puname,$pudom,$prole,$adj, + &escape($requesthash{$item}{'requester'}), + $psec); + } elsif ($item =~ /^status&/) { + my ($dummy,$key) = split(/&/,$item,2); + $reqstatus{$key} = $requesthash{$item}; + } + } elsif ($context eq 'othdomaction') { + next unless ($item =~ /^pending:/); + if (ref($requesthash{$item}) eq 'HASH') { + $timestamp = $requesthash{$item}{'timestamp'}; + $entry = &escape($item).':'.&escape($requesthash{$item}{'requester'}); + } } else { - $timestamp = $requesthash{$item}{'timestamp'}; if (ref($requesthash{$item}) eq 'HASH') { + $timestamp = $requesthash{$item}{'timestamp'}; my ($cnum,$disposition) = split('_',$item); $entry = $cnum.':'.$requesthash{$item}{'ownername'}.':'. - $requesthash{$item}{'ownerdom'}.':'. - $requesthash{$item}{'crstype'}.':'. - $requesthash{$item}{'description'}; + $requesthash{$item}{'ownerdom'}.':'; + if (($context eq 'pending') || ($context eq 'displaypending')) { + $entry .= $requesthash{$item}{'instcode'}; + } else { + $entry .= $requesthash{$item}{'crstype'}; + } + $entry .= ':'.$requesthash{$item}{'description'}; } } if ($entry ne '') { - if (exists($queue_by_date{$timestamp})) { - if (ref($queue_by_date{$timestamp}) eq 'ARRAY') { - push(@{$queue_by_date{$timestamp}},$entry); - } - } else { - @{$queue_by_date{$timestamp}} = ($entry); + if (ref($queue_by_date{$timestamp}) eq 'ARRAY') { + push(@{$queue_by_date{$timestamp}},$entry); + } else { + $queue_by_date{$timestamp} = [$entry]; } } } - my @sortedtimes = sort {$a <=> $b} (keys(%queue_by_date)); - my $count = 0; - foreach my $item (@sortedtimes) { - if (ref($queue_by_date{$item}) eq 'ARRAY') { - foreach my $request (sort(@{$queue_by_date{$item}})) { - my ($row,$approve,$reject,$showtime,$showsec,$namelink, - $detailslink,$crstype); - $showtime = &Apache::lonlocal::locallocaltime($item); - if ($context eq 'course') { - my ($puname,$pudom,$pusec) = split(/:/,$request); - $approve = $count.':'.$puname.':'.$pudom.':'.$pusec; - $reject = $puname.':'.$pudom; - $showsec = $pusec; - if ($showsec eq '') { - $showsec = &mt('none'); - } - $namelink = &Apache::loncommon::aboutmewrapper( - &Apache::loncommon::plainname($puname,$pudom), - $puname,$pudom); + } + if (keys(%queue_by_date) > 0) { + $output = '<form method="post" name="changequeue" action="'.$formaction.'" />'."\n". + '<input type="hidden" name="action" value="'.$env{'form.action'}.'" />'."\n". + $nextelement."\n"; + if ($context eq 'course') { + $output .= '<h3>'.&mt('Self-enrollment requests queued pending approval by a Coordinator').'</h3>'; + } elsif (($context eq 'pending') || ($context eq 'displaypending')) { + $output .= '<h3>'.&mt('Requests for official courses queued pending validation').'</h3>'. + '<p>'.&mt('Requests are validated against institutional data to confirm that the requestor is an instructor of record.').'<br />'. + &mt('Validation is attempted when the request is submitted.').' '. + &mt('If unvalidated, the request will be held in a queue.').' '. + &mt('Validation of pending requests is automatically repeated daily.').'</p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<h3>'.&mt('Requests for Authoring Space queued pending approval by a Domain Coordinator').'</h3>'; + } elsif ($context eq 'requestusername') { + $output .= '<h3>'.&mt('Requests for LON-CAPA accounts queued pending approval by a Domain Coordinator').'</h3>'; + } elsif ($context eq 'othdomqueue') { + if ($secondary eq 'domain') { + $output .= '<h3>'.&mt('Domain role assignments for users from another domain which were/are queued for approval').'</h3>'; + } elsif ($secondary eq 'author') { + $output .= '<h3>'.&mt('Co-author role assignments for users from another domain which were/are queued for approval').'</h3>'; + } elsif ($secondary eq 'course') { + $output .= '<h3>'.&mt('Course role assignments for users from another domain which were/are queued for approval').'</h3>'; + } elsif ($secondary eq 'community') { + $output .= '<h3>'.&mt('Community role assignments for users from another domain which were/are queued for approval').'</h3>'; + } + } elsif ($context eq 'othdomaction') { + unless ($secondary eq 'user') { + $output .= '<h3>'.&mt('Role assignments in other domains, queued pending domain coordinator approval in this domain.').'</h3>'; + } + } else { + $output .= '<h3>'.&mt('Course/Community requests queued pending approval by a Domain Coordinator').'</h3>'; + } + if ($context eq 'othdomqueue') { + $output .= &queued_role_display($secondary,\%queue_by_date,\%reqstatus); + } else { + $output .= &build_queue_display($dom,$context,\%queue_by_date,$secondary). + '<input type="hidden" name="queue" value="approval" />'; + if ($secondary eq 'user') { + $output .= "\n".'<input type="hidden" name="approvals" value="show" />'."\n"; + } + } + if ($context eq 'pending') { + $output .= '<br /><input type="submit" name="validationcheck" value="'. + &mt('Validate').'" /><br />'."\n". + '<p>'.&mt('Any course/community requests which are successfully validated will be created immediately.').' '. + &mt('Unvalidated requests will be listed for manual approval/rejection.').'</p>'; + } elsif (($context ne 'helpdesk') && ($context ne 'displaypending') && ($context ne 'othdomqueue')) { + $output .= '<br /><input type="submit" name="processqueue" value="'.&mt('Save').'" />'; + } + $output .= '</form>'; + } else { + if (($context eq 'othdomaction') && ($secondary eq 'user')) { + $output .= '<span class="LC_info">'; + } else { + $output .= '<div class="LC_info">'; + } + if ($context eq 'course') { + $output .= &mt('There are currently no enrollment requests awaiting approval.'); + } elsif (($context eq 'pending') || ($context eq 'displaypending')) { + $output .= &mt('There are currently no requests for official courses awaiting validation.'); + } elsif ($context eq 'requestauthor') { + $output .= &mt('There are currently no requests for Authoring Space awaiting approval.'); + } elsif ($context eq 'requestusername') { + $output .= &mt('There are currently no requests for LON-CAPA accounts awaiting approval.'); + } elsif ($context eq 'othdomqueue') { + if ($secondary eq 'domain') { + $output .= &mt('There are currently no domain role assignment(s) for user(s) from another domain which were/are queued for approval'); + } elsif ($secondary eq 'author') { + $output .= &mt('There are currently no co-author role assignment(s) for user(s) from another domain which were/are queued for approval'); + } elsif ($secondary eq 'course') { + $output .= &mt('There are currently no course role assignment(s) for user(s) from another domain which were/are queued for approval'); + } elsif ($secondary eq 'community') { + $output .= &mt('There are currently no community role assignment(s) for user(s) from another domain which were/are queued for approval'); + } + } elsif ($context eq 'othdomaction') { + if ($secondary eq 'user') { + $output .= &mt('No role assignments for you in other domains currently awaiting your acceptance'); + } elsif ($secondary eq 'domain') { + $output .= &mt('No role assignments in other domains currently awaiting domain coordinator approval'); + } + } else { + $output .= &mt('There are currently no course or community requests awaiting approval.'); + } + if (($context eq 'othdomaction') && ($secondary eq 'user')) { + $output .= '</span>'; + } else { + $output .= '</div>'; + } + } + return $output; +} +sub build_queue_display { + my ($dom,$context,$queue,$secondary) = @_; + return unless (ref($queue) eq 'HASH'); + my (%crstypes,%roles_by_context,$output); + $output .= &Apache::loncommon::start_data_table(). + &Apache::loncommon::start_data_table_header_row(); + unless (($context eq 'pending') || ($context eq 'displaypending') || + ($context eq 'helpdesk')) { + $output .= '<th>'.&mt('Action').'</th>'; + } + unless (($context eq 'othdomaction') && ($secondary eq 'user')) { + $output .= '<th>'.&mt('Requestor').'</th>'; + } + if ($context eq 'course') { + $output .= '<th>'.&mt('Section').'</th>'. + '<th>'.&mt('Date requested').'</th>'; + } elsif ($context eq 'requestauthor') { + $output .= '<th>'.&mt('Date requested').'</th>'; + } elsif ($context eq 'requestusername') { + $output .= '<th>'.&mt('Date requested').'</th>'. + '<th>'.&mt('Details').'</th>'; + } elsif ($context eq 'othdomaction') { + $output .= '<th>'.&mt('Date requested').'</th>'. + '<th>'.&mt('Role type').'</th>'. + '<th>'.&mt('Location').'</th>'; + if ($secondary eq 'domain') { + $output .= '<th>'.&mt('Affected User').'</th>'; + } + foreach my $type ('domain','course') { + my @possroles = &Apache::lonuserutils::roles_by_context($type); + $roles_by_context{$type} = \@possroles; + } + } elsif ($context eq 'pending' || $context eq 'displaypending' || $context eq 'stillpending') { + $output .= '<th>'.&mt('Institutional code').'</th>'. + '<th>'.&mt('Date requested').'</th>'. + '<th>'.&mt('Details').'</th>'; + } else { + %crstypes = &Apache::lonlocal::texthash ( + official => 'Official course', + unofficial => 'Unofficial course', + community => 'Community', + textbook => 'Textbook course', + placement => 'Placement test', + ); + $output .= '<th>'.&mt('Type').'</th>'. + '<th>'.&mt('Date requested').'</th>'. + '<th>'.&mt('Details').'</th>'; + } + $output .= &Apache::loncommon::end_data_table_header_row(); + my @sortedtimes = sort {$a <=> $b} (keys(%{$queue})); + my $count = 0; + foreach my $item (@sortedtimes) { + if (ref($queue->{$item}) eq 'ARRAY') { + foreach my $request (sort(@{$queue->{$item}})) { + my ($row,$approve,$reject,$showtime,$showsec,$namelink, + $detailslink,$crstype,$instcode,$showrole,$adjudicator, + $location,$showrequester); + $showtime = &Apache::lonlocal::locallocaltime($item); + if ($context eq 'course') { + my ($puname,$pudom,$pusec) = split(/:/,$request); + $approve = $count.':'.$puname.':'.$pudom.':'.$pusec; + $reject = $puname.':'.$pudom; + $showsec = $pusec; + if ($showsec eq '') { + $showsec = &mt('none'); + } + $namelink = &Apache::loncommon::aboutmewrapper( + &Apache::loncommon::plainname($puname,$pudom), + $puname,$pudom); + } elsif ($context eq 'requestauthor') { + if (&Apache::lonnet::homeserver($request,$dom) ne 'no_host') { + $approve = $count.':'.$request; + $reject = $request; + $namelink = &Apache::loncommon::aboutmewrapper( + &Apache::loncommon::plainname($request,$dom), + $request,$dom); + } + } elsif ($context eq 'requestusername') { + if (&Apache::lonnet::homeserver($request,$dom) eq 'no_host') { + my $queued = 'approval'; + $approve = $count.':'.$request; + $reject = $request; + $detailslink='<a href="javascript:openusernamereqdisplay('. + "'$dom','$request','$queued'".');">'.$request.'</a>'; + $namelink = $request; + } + } elsif ($context eq 'othdomaction') { + my ($status,$extent,$role,$crstype); + my ($info,$requester) = map { &unescape($_); } split(/:/,$request); + if ($secondary eq 'user') { + ($status,$extent,$role) = split(/:/,$info); + $approve = $count.':'.$extent.':'.$role; + $reject = $extent.':'.$role; + } elsif ($secondary eq 'domain') { + ($status,my $uname,$extent,$role) = split(/:/,$info); + $approve = $count.':'.$extent.':'.$role.':'.$uname.':'.$dom; + $reject = $extent.':'.$role.':'.$uname.':'.$dom; + unless (&Apache::lonnet::homeserver($uname,$dom) eq 'no_host') { + $namelink = &Apache::loncommon::plainname($uname,$dom); + unless ($namelink eq $uname.':'.$dom) { + $namelink .= ' ('.$uname.':'.$dom.')'; + } + } + } + if (($role eq 'ca') || ($role eq 'aa')) { + my ($audom,$auname) = ($extent =~ m{^/($match_domain)/($match_username)$}); + $location = &mt('Domain').': '.&Apache::lonnet::domain($audom,'description').'<br />'. + &mt('Author').': '.&Apache::loncommon::plainname($auname,$audom); + } elsif ($role eq 'co') { + my ($cdom,$cnum) = ($extent =~ m{^/($match_domain)/($match_courseid)}); + if (&Apache::lonnet::is_course($cdom,$cnum)) { + my %info = &Apache::lonnet::coursedescription("$cdom/$cnum",{'one_time' => 1}); + $crstype = $info{'type'}; + $location = &mt('Domain').': '.&Apache::lonnet::domain($cdom,'description').'<br />'. + &mt('Community').': '.$info{'description'}; + $showrole = &Apache::lonnet::plaintext($role,'Community'); + } + } elsif ($role =~ m{^cr/}) { + $showrole = &Apache::lonnet::plaintext($role,$crstype); + my ($cdom,$cnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/(\w+))$}); + if (&Apache::lonnet::is_course($cdom,$cnum)) { + my %info = &Apache::lonnet::coursedescription("$cdom/$cnum",{'one_time' => 1}); + $crstype = $info{'type'}; + $location = &mt('Domain').': '.&Apache::lonnet::domain($cdom,'description').'<br />'. + &mt($crstype).': '.$info{'description'}; + if ($csec ne '') { + $location .= '<br />'.&mt('Section').': '.$csec; + } + } } else { - my ($cnum,$ownername,$ownerdom,$type,$cdesc)=split(/:/,$request,5); - $detailslink='<a href="javascript:opencoursereqdisplay('. - "'$dom','$cnum'".');">'.$cdesc.'</a>'; + foreach my $type ('course','domain') { + if (grep(/^\Q$role\E$/,@{$roles_by_context{$type}})) { + if ($type eq 'course') { + my ($cdom,$cnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/(\w+))$}); + if (&Apache::lonnet::is_course($cdom,$cnum)) { + my %info = &Apache::lonnet::coursedescription("$cdom/$cnum",{'one_time' => 1}); + $crstype = $info{'type'}; + $location = &mt('Domain').': '.&Apache::lonnet::domain($cdom,'description').'<br />'. + &mt($crstype).': '.$info{'description'}; + if ($csec ne '') { + $location .= '<br />'.&mt('Section').': '.$csec; + } + } + } else { + my ($domain) = ($extent =~ m{^/($match_domain)/$}); + $location = &mt('Domain').': '.&Apache::lonnet::domain($domain,'description'); + } + last; + } + } + } + if ($role =~ m{^cr/($match_domain)/($match_username)/(\w+)$}) { + my ($crudom,$cruname,$rolename) = ($1,$2,$3); + my $creator = &Apache::loncommon::plainname($cruname,$crudom); + unless ($creator eq $cruname.':'.$crudom) { + $creator .= ' ('.$cruname.':'.$crudom.')'; + } + $showrole = &mt('Custom role').': '.$rolename.'<br />'. + &mt('Created by').' '.$creator; + } else { + $showrole = &Apache::lonnet::plaintext($role,$crstype); + } + my ($requname,$requdom) = split(/:/,$requester); + $showrequester = &Apache::loncommon::plainname($requname,$requdom); + unless ($showrequester eq $requname.':'.$requdom) { + $showrequester .= ' ('.$requname.':'.$requdom.')'; + } + } else { + my ($cnum,$ownername,$ownerdom,$type,$cdesc); + my $queued = 'approval'; + if ($context eq 'pending' || $context eq 'displaypending' || $context eq 'stillpending') { + ($cnum,$ownername,$ownerdom,$instcode,$cdesc)=split(/:/,$request,5); + $queued = 'pending'; + } else { + ($cnum,$ownername,$ownerdom,$type,$cdesc)=split(/:/,$request,5); $crstype = $type; if (defined($crstypes{$type})) { $crstype = $crstypes{$type}; } - $approve = $count.':'.$cnum; - $reject = $cnum; - $namelink = &Apache::loncommon::aboutmewrapper( - &Apache::loncommon::plainname($ownername,$ownerdom), - $ownername,$ownerdom); } + $detailslink='<a href="javascript:opencoursereqdisplay('. + "'$dom','$cnum','$queued'".');">'.$cdesc.'</a>'; + $approve = $count.':'.$cnum; + $reject = $cnum; + $namelink = &Apache::loncommon::aboutmewrapper( + &Apache::loncommon::plainname($ownername,$ownerdom), + $ownername,$ownerdom); + } + unless (($context eq 'pending') || ($context eq 'displaypending') || + ($context eq 'helpdesk')) { $row = '<td><span class="LC_nobreak"><label>'. - '<input type="checkbox" value="'.$approve.'" name="approvereq" />'.&mt('Approve').'</label></span><br />'. - '<span class="LC_nobreak"><label>'. - '<input type="checkbox" value="'.$reject.'" name="rejectreq" />'.&mt('Reject').'</label></span><br /></td>'. - '<td>'.$namelink.'</td>'."\n"; - if ($context eq 'course') { - $row .= '<td>'.$showsec.'</td>'."\n". - '<td>'.$showtime.'</td>'."\n"; - } else { - $row .= '<td>'.$crstype.'</td>'."\n". - '<td>'.$showtime.'</td>'."\n". - '<td>'.$detailslink.'</td>'."\n"; - } - $output .= &Apache::loncommon::start_data_table_row()."\n". - $row. - &Apache::loncommon::end_data_table_row()."\n"; - $count ++; + '<input type="radio" value="'.$approve.'" name="'.$count.'radioreq" />'.&mt('Approve').'</label>'. + '<label>'.(' 'x2). + '<input type="radio" value="'.$reject.'" name="'.$count.'radioreq" />'.&mt('Reject').'</label>'. + '<label>'.(' 'x2). + '<input type="radio" value="'."later:".$reject.'" name="'.$count.'radioreq" checked />'.&mt('Decide Later'). + '</label></span><br /></td>'; + } + unless ($context eq 'othdomaction') { + $row .= '<td>'.$namelink.'</td>'."\n"; } + if ($context eq 'course') { + $row .= '<td>'.$showsec.'</td>'."\n". + '<td>'.$showtime.'</td>'."\n"; + } elsif ($context eq 'requestauthor') { + $row .= '<td>'.$showtime.'</td>'."\n"; + } elsif ($context eq 'requestusername') { + $row .= '<td>'.$showtime.'</td>'."\n". + '<td>'.$detailslink.'</td>'."\n"; + } elsif ($context eq 'othdomaction') { + if ($secondary eq 'domain') { + $row .= '<td>'.$showrequester.'</td>'."\n"; + } + $row .= '<td>'.$showtime.'</td>'."\n". + '<td>'.$showrole.'</td>'."\n". + '<td>'.$location.'</td>'."\n"; + if ($secondary eq 'domain') { + $row .= '<td>'.$namelink.'</td>'."\n"; + } + } else { + if ($context eq 'pending' || $context eq 'displaypending' || $context eq 'stillpending') { + $row .= '<td>'.$instcode.'</td>'."\n"; + } else { + $row .= '<td>'.$crstype.'</td>'."\n"; + } + $row .= '<td>'.$showtime.'</td>'."\n". + '<td>'.$detailslink.'</td>'."\n"; + } + $output .= &Apache::loncommon::start_data_table_row()."\n". + $row. + &Apache::loncommon::end_data_table_row()."\n"; + $count ++; } } - $output .= &Apache::loncommon::end_data_table(). - '<input type="submit" name="processqueue" value="'.&mt('Save'). - '" /></form>'; + } + $output .= &Apache::loncommon::end_data_table(); + return $output; +} + +sub queued_role_display { + my ($context,$queue,$status) = @_; + return unless ((ref($queue) eq 'HASH') && (ref($status) eq 'HASH')); + my (%curr,$minshown,$maxshown,$more_records,$crstype,$viewablesec,$output); + my $formname = 'changequeue'; + if ($context eq 'course') { + $crstype = &Apache::loncommon::course_type(); + my ($permission,$allowed) = + &Apache::lonuserutils::get_permission($context,$crstype); + $viewablesec = &Apache::lonuserutils::viewable_section($permission); + my %saveable_parameters = ('show' => 'scalar',); + &Apache::loncommon::store_course_settings('roles_req', + \%saveable_parameters); + &Apache::loncommon::restore_course_settings('roles_req', + \%saveable_parameters); + } + +# Create navigation javascript + my $jsnav = &queued_log_js($formname); + + $output = (<<ENDSCRIPT); +<script type="text/javascript"> +// <![CDATA[ +$jsnav +// ]]> +</script> +ENDSCRIPT + + my $now = time(); + my $defstart = $now - (7*24*3600); #7 days ago + my %defaults = ( + page => '1', + show => '10', + role => 'any', + chgstatus => 'any', + chgadj => 'any', + rolereq_start_date => $defstart, + rolereq_end_date => $now, + ); + $more_records = 0; + my %lt = &othdomrole_contexts(); + + foreach my $item ('show','page','role','chgstatus','chgadj') { + $curr{$item} = $env{'form.'.$item}; + } + ($curr{'rolereq_start_date'},$curr{'rolereq_end_date'}) = + &Apache::lonuserutils::get_dates_from_form('rolereq_start_date','rolereq_end_date'); + foreach my $key (keys(%defaults)) { + if ($curr{$key} eq '') { + $curr{$key} = $defaults{$key}; + } + } + $minshown = 1; + my $count = 0; + if ($curr{'show'} =~ /\D/) { + $curr{'page'} = 1; } else { - if ($context eq 'course') { - $output .= &mt('There are currently no enrollment requests.'); - } else { - $output .= &mt('There are currently no course requests awaiting approval.'); + $maxshown = $curr{'page'} * $curr{'show'}; + if ($curr{'page'} > 1) { + $minshown = 1 + ($curr{'page'} - 1) * $curr{'show'}; } } + $output .= &print_filter_menu($context,'changequeue',\%curr,$crstype); + + my $showntableheader = 0; + + # Table Header + my $tableheader = + &Apache::loncommon::start_data_table(). + &Apache::loncommon::start_data_table_header_row(). + '<th>'.&mt('Date requested').'</th>'. + '<th>'.&mt('Role').'</th>'; + if ($context eq 'course') { + $tableheader .= '<th>'.&mt('Section').'</th>'; + } + $tableheader .= '<th>'.&mt('Requested for').'</th>'. + '<th>'.&mt('Request status').'</th>'. + '<th>'.&mt('Adjudicator').'</th>'. + &Apache::loncommon::end_data_table_header_row(); + + my @sortedtimes = sort {$a <=> $b} (keys(%{$queue})); + my $count = 0; + foreach my $item (@sortedtimes) { + next if (($item < $curr{'rolereq_start_date'}) || + ($item > $curr{'rolereq_end_date'})); + if (ref($queue->{$item}) eq 'ARRAY') { + foreach my $request (sort(@{$queue->{$item}})) { + if ($curr{'show'} !~ /\D/) { + if ($count >= $curr{'page'} * $curr{'show'}) { + $more_records = 1; + last; + } + } + my ($showtime,$showsec,$namelink,$showrole,$showadj, + $showstatus,$id); + $showtime = &Apache::lonlocal::locallocaltime($item); + my ($uname,$udom,$role,$adj,$requester,$sec) = split(/:/,$request); + $id = join(':',($uname,$udom,$role)); + if ($context eq 'course') { + $id .= ':'.$sec; + } + if ($curr{'role'} ne 'any') { + if ($curr{'role'} eq 'cr') { + next unless ($role =~ m{^cr/}); + } else { + next unless ($role eq $curr{'role'}); + } + } + if ($curr{'chgstatus'} ne 'any') { + next if ($status->{$id} ne $curr{'chgstatus'}); + } + if ($curr{'chgadj'} ne 'any') { + next if ($adj ne $curr{'chgadj'}); + } + if (($context eq 'course') && ($viewablesec ne '')) { + next if ($sec ne $viewablesec); + } + $count ++; + next if ($count < $minshown); + unless ($showntableheader) { + $output .= $tableheader; + $showntableheader = 1; + } + $showrole = &Apache::lonnet::plaintext($role,$crstype); + $showstatus = $lt{$status->{$id}}; + $showadj = $lt{$adj}; + unless (&Apache::lonnet::homeserver($uname,$udom) eq 'no_host') { + $namelink = &Apache::loncommon::plainname($uname,$udom)." ($uname:$udom)"; + } + if ($context eq 'course') { + $showsec = $sec; + if ($showsec eq '') { + $showsec = &mt('none'); + } + } + $output .= &Apache::loncommon::start_data_table_row()."\n". + '<td>'.$showtime.'</td>'."\n". + '<td>'.$showrole.'</td>'."\n"; + if ($context eq 'course') { + $output .= '<td>'.$showsec.'</td>'."\n"; + } + $output .= '<td>'.$namelink.'</td>'."\n". + '<td>'.$showstatus.'</td>'."\n". + '<td>'.$showadj.'</td>'."\n". + &Apache::loncommon::end_data_table_row()."\n"; + } + } + } + + if ($showntableheader) { # Table footer, if content displayed above + $output .= &Apache::loncommon::end_data_table(). + &queued_role_navlinks(\%curr,$more_records); + } else { # No content displayed above + $output .= '<p class="LC_info">'. + &mt('There are no records to display.'). + '</p>'; + } + $output .= '<input type="hidden" name="page" value="'.$curr{'page'}.'" />'; return $output; } +sub queued_log_js { + my ($formname) = @_; + return <<"ENDSCRIPT"; + +function chgPage(caller) { + if (caller == 'previous') { + document.$formname.page.value --; + } + if (caller == 'next') { + document.$formname.page.value ++; + } + document.$formname.submit(); + return; +} +ENDSCRIPT +} + +sub queued_role_navlinks { + my ($curr,$more_records) = @_; + return unless(ref($curr) eq 'HASH'); + # Navigation Buttons + my $nav_links; + if (($curr->{'page'} > 1) || ($more_records)) { + $nav_links = '<p>'; + if (($curr->{'page'} > 1) && ($curr->{'show'} !~ /\D/)) { + $nav_links .= '<input type="button"' + .' onclick="javascript:chgPage('."'previous'".');"' + .' value="'.&mt('Previous [_1] changes',$curr->{'show'}) + .'" /> '; + } + if ($more_records) { + $nav_links .= '<input type="button"' + .' onclick="javascript:chgPage('."'next'".');"' + .' value="'.&mt('Next [_1] changes',$curr->{'show'}) + .'" />'; + } + $nav_links .= '</p>'; + } + return $nav_links; +} + +sub print_filter_menu { + my ($context,$formname,$curr,$crstype) = @_; + + my $nolink = 1; + my $output = '<table><tr><td valign="top">'. + '<span class="LC_nobreak"><b>'.&mt('Changes/page:').'</b></span><br />'. + &Apache::lonmeta::selectbox('show',$curr->{'show'},'','',undef, + (&mt('all'),5,10,20,50,100,1000,10000)). + '</td><td> </td>'; + my $startform = + &Apache::lonhtmlcommon::date_setter($formname,'rolereq_start_date', + $curr->{'rolereq_start_date'},undef, + undef,undef,undef,undef,undef,undef,$nolink); + my $endform = + &Apache::lonhtmlcommon::date_setter($formname,'rolereq_end_date', + $curr->{'rolereq_end_date'},undef, + undef,undef,undef,undef,undef,undef,$nolink); + my %lt = &othdomrole_contexts(); + $output .= '<td valign="top"><b>'.&mt('Time window in which role was requested').':</b><br />'. + '<table><tr><td>'.&mt('After:'). + '</td><td>'.$startform.'</td></tr>'. + '<tr><td>'.&mt('Before:').'</td>'. + '<td>'.$endform.'</td></tr></table>'. + '</td>'. + '<td> </td>'. + '<td valign="top"><b>'.&mt('Requested role').':</b><br />'. + '<select name="role"><option value="any"'; + if ($curr->{'role'} eq 'any') { + $output .= ' selected="selected"'; + } + $output .= '>'.&mt('Any').'</option>'."\n"; + my @roles = &Apache::lonuserutils::roles_by_context($context,1,$crstype); + foreach my $role (@roles) { + my $plrole; + if ($role eq 'cr') { + $plrole = &mt('Custom Role'); + } else { + $plrole=&Apache::lonnet::plaintext($role,$crstype); + } + my $selstr = ''; + if ($role eq $curr->{'role'}) { + $selstr = ' selected="selected"'; + } + $output .= ' <option value="'.$role.'"'.$selstr.'>'.$plrole.'</option>'; + } + $output .= '</select></td>'. + '<td> </td>'. + '<td valign="top"><b>'. + &mt('Request status').':</b><br />'. + '<select name="chgstatus">'. + '<option value="any"'; + if ($curr->{'chgstatus'} eq 'any') { + $output .= ' selected="selected"'; + } + $output .= '>'.&mt('Any').'</option>'."\n"; + my @possstatus = ('pending','approved','rejected'); + foreach my $statustype (@possstatus) { + my $selstr = ''; + if ($curr->{'chgstatus'} eq $statustype) { + $selstr = ' selected="selected"'; + } + $output .= '<option value="'.$statustype.'"'.$selstr.'>'.$lt{$statustype}.'</option>'."\n"; + } + $output .= '</select></td>'. + '<td> </td>'. + '<td valign="top"><b>'. + &mt('Adjudicator').':</b><br />'. + '<select name="chgadj">'. + '<option value="any"'; + if ($curr->{'adj'} eq 'any') { + $output .= ' selected="selected"'; + } + $output .= '>'.&mt('Any').'</option>'."\n"; + my @possadj = ('domain','user'); + foreach my $adjtype (@possadj) { + my $selstr = ''; + if ($curr->{'chgadj'} eq $adjtype) { + $selstr = ' selected="selected"'; + } + $output .= '<option value="'.$adjtype.'"'.$selstr.'>'.$lt{$adjtype}.'</option>'."\n"; + } + $output .= '</select></td>' + .'</tr></table>'; + + # Update Display button + $output .= '<p>'. + '<input type="submit" value="'.&mt('Update Display').'" />'. + '</p>'. + '<hr />'; + return $output; +} + +sub othdomrole_contexts { + my %lt = &Apache::lonlocal::texthash( + pending => 'Queued', + approved => 'Approved', + rejected => 'Rejected', + user => 'User who acquires role', + ); + $lt{'domain'} = &mt("[_1] in user's domain", + &Apache::lonnet::plaintext('dc')); + return %lt; +} + sub update_request_queue { my ($context,$cdom,$cnum,$coursedesc) = @_; my ($output,$access_start,$access_end,$limit,$cap,$notifylist,$namespace, - $stucounts,$idx,$classlist,%requesthash,$cid,$hostname,$protocol, - $domdesc,$now,$sender,$approvedmsg,$rejectedmsg,$beneficiary, + $stucounts,$idx,$classlist,%requesthash,$cid,$domdesc,$now, + $sender,$approvedmsg,$rejectedmsg,$beneficiary, @existing,@missingreq,@invalidusers,@limitexceeded,@completed, - @processing_errors,@warn_approves,@warn_rejects,@approvals, - @rejections,@rejectionerrors,@nopermissions,%courseroles, - %communityroles,%domdefs,@warn_coursereqs); - @approvals = &Apache::loncommon::get_env_multiple('form.approvereq'); - @rejections = &Apache::loncommon::get_env_multiple('form.rejectreq'); + @processing_errors,@warn_approves,@warn_rejects,@approvals,@warn_dels, + @rejections,@rejectionerrors,@nopermissions,%courseroles,@toremove, + %communityroles,%domdefs,%approvalmsg,%rejectionmsg,$crstype,$queue, + $firsturl,$uniquecode,%codes,%roles_by_context,%requesteractive, + %gotroles,$confname,%requestedby,@rejectedreqs,$dbname); + my $count=0; + while (my $item = $env{'form.'.$count.'radioreq'}) { + if ($item =~ /^\d+:/) { + push(@approvals,$item); + } elsif ($item !~ /^later:/) { + push(@rejections,$item); + } + $count ++; + } $now = time; $sender = $env{'user.name'}.':'.$env{'user.domain'}; if ($context eq 'course') { $namespace = 'selfenrollrequests'; $beneficiary = 'enroller'; $cid = $env{'request.course.id'}; - my $chome = &Apache::lonnet::homeserver($cnum,$cdom); - $hostname = &Apache::lonnet::hostname($chome); - $protocol = $Apache::lonnet::protocol{$chome}; - $protocol = 'http' if ($protocol ne 'https'); + $crstype = lc(&Apache::loncommon::course_type()); + $firsturl = &Apache::lonnet::course_portal_url($cnum,$cdom); %requesthash = &Apache::lonnet::dump($namespace,$cdom,$cnum); $access_start = $env{'course.'.$cid.'.internal.selfenroll_start_access'}; $access_end = $env{'course.'.$cid.'.internal.selfenroll_end_access'}; @@ -370,37 +1211,117 @@ sub update_request_queue { mt => 'Your request for enrollment has been approved.', }, { - mt => 'Visit [_1], to log-in and access the course', - args => [$protocol.'://'.$hostname], + mt => 'Visit [_1] to log-in and access the course', + args => [$firsturl], }]; $rejectedmsg = [{ mt => 'Your request for enrollment has not been approved.', }]; + } elsif ($context eq 'requestauthor') { + $namespace = 'requestauthorqueue'; + $beneficiary = 'requestauthor'; + %requesthash = &Apache::lonnet::dump_dom($namespace,$cdom); + my %domdefs = &Apache::lonnet::get_domain_defaults($cdom); + if (ref($domdefs{'requestauthor'}) eq 'HASH') { + if (ref($domdefs{'requestauthor'}{'notify'}) eq 'HASH') { + $notifylist = $domdefs{'requestauthor'}{'notify'}{'approval'}; + } + } + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + $firsturl = &Apache::lonnet::course_portal_url($domconfiguser,$cdom); + $approvedmsg = [{ + mt => 'Your request for Authoring Space has been approved.', + }, + { + mt => 'Visit [_1] to log-in and select your author role', + args => [$firsturl], + }]; + $rejectedmsg = [{ + mt => 'Your request for Authoring Space has not been approved.', + }]; + $domdesc = &Apache::lonnet::domain($cdom); + } elsif ($context eq 'requestusername') { + $namespace = 'usernamequeue'; + $beneficiary = 'requestusername'; + %requesthash = &Apache::lonnet::dump_dom($namespace,$cdom); + my %domconfig = &Apache::lonnet::get_dom('configuration',['usercreation'],$cdom); + if (ref($domconfig{'usercreation'}) eq 'HASH') { + if (ref($domconfig{'usercreation'}{'cancreate'}) eq 'HASH') { + if (ref($domconfig{'usercreation'}{'cancreate'}{'notify'}) eq 'HASH') { + $notifylist = $domconfig{'usercreation'}{'cancreate'}{'notify'}{'approval'}; + } + } + } + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + $firsturl = &Apache::lonnet::course_portal_url($domconfiguser,$cdom); + $approvedmsg = [{ + mt => 'Your request for a LON-CAPA account has been approved.', + }, + { + mt => 'Visit [_1] to log-in.', + args => [$firsturl], + }]; + $rejectedmsg = [{ + mt => 'Your request for a LON-CAPA account has not been approved.', + }]; + $domdesc = &Apache::lonnet::domain($cdom); + $dbname = 'nohist_requestedusernames'; + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + $namespace = 'nohist_queuedrolereqs'; + foreach my $type ('domain','course') { + my @possroles = &Apache::lonuserutils::roles_by_context($type); + $roles_by_context{$type} = \@possroles; + } + if ($context eq 'othdombydc') { + $confname = &Apache::lonnet::get_domainconfiguser($cdom); + %requesthash = &Apache::lonnet::dump($namespace,$cdom,$confname); + } elsif ($context eq 'othdombyuser') { + %requesthash = &Apache::lonnet::dump($namespace); + } + $domdesc = &Apache::lonnet::domain($cdom); + $dbname = 'nohist_othdomqueued'; } else { $domdesc = &Apache::lonnet::domain($cdom); $namespace = 'courserequestqueue'; $beneficiary = 'courserequestor'; - %requesthash = &Apache::lonnet::dump_dom($namespace,$cdom,'_approval'); - my $chome = &Apache::lonnet::domain($cdom,'primary'); - $hostname = &Apache::lonnet::hostname($chome); - $protocol = $Apache::lonnet::protocol{$chome}; - $protocol = 'http' if ($protocol ne 'https'); + $queue = 'approval'; + if ($env{'form.queue'} eq 'pending') { + $queue = 'pending'; + } + %requesthash = &Apache::lonnet::dump_dom($namespace,$cdom,'_'.$queue); my %domconfig = &Apache::lonnet::get_dom('configuration',['requestcourses'],$cdom); if (ref($domconfig{'requestcourses'}) eq 'HASH') { if (ref($domconfig{'requestcourses'}{'notify'}) eq 'HASH') { $notifylist = $domconfig{'requestcourses'}{'notify'}{'approval'}; } } - $approvedmsg = [{ + $approvalmsg{'course'} = + [{ mt => 'Your course request has been approved.', }, { - mt => 'Visit [_1], to log-in and access the course', - args => [$protocol.'://'.$hostname], + mt => 'Visit [_1] to log-in and access the course', + args => [], }]; - $rejectedmsg = [{ + $rejectionmsg{'course'} = + [{ mt => 'Your course request has not been approved.', }]; + + $approvalmsg{'community'} = + [{ + mt => 'Your community request has been approved.', + }, + { + mt => 'Visit [_1] to log-in and access the community', + args => [], + }]; + + $rejectionmsg{'community'} = + [{ + mt => 'Your community request has not been approved.', + }]; + %domdefs = &Apache::lonnet::get_domain_defaults($cdom); my @roles = &Apache::lonuserutils::roles_by_context('course'); foreach my $role (@roles) { @@ -409,7 +1330,6 @@ sub update_request_queue { foreach my $role (@roles) { $communityroles{$role}=&Apache::lonnet::plaintext($role,'Community'); } - } foreach my $item (sort {$a <=> $b} @approvals) { if ($context eq 'course') { @@ -417,7 +1337,6 @@ sub update_request_queue { my $uhome = &Apache::lonnet::homeserver($uname,$udom); if ($uhome ne 'no_host') { if (exists($requesthash{$uname.':'.$udom})) { - if (exists($classlist->{$uname.':'.$udom})) { if (ref($classlist->{$uname.':'.$udom}) eq 'ARRAY') { if (($classlist->{$uname.':'.$udom}->[$idx->{'status'}] eq 'Active') || @@ -450,7 +1369,8 @@ sub update_request_queue { $stucounts->{'allstudents'} ++; $stucounts->{'selfenrolled'} ++; &send_selfserve_notification($uname.':'.$udom,$approvedmsg, - $cid,$coursedesc,$now,$beneficiary,$sender); + $cid,$coursedesc,$now,$beneficiary,$sender, + undef,undef,$crstype); my %userrequest = ( $cdom.'_'.$cnum => { timestamp => $now, @@ -463,15 +1383,6 @@ sub update_request_queue { &Apache::lonnet::put($namespace,\%userrequest,$udom,$uname); if ($userresult ne 'ok') { push(@warn_approves,$uname.':'.$udom); - } elsif ($udom eq 'gci') { - my %changehash = ( - 'reqcrsotherdom.unofficial' => 'gcitest:autolimit=', - ); - my $reqresult = &Apache::lonnet::put('environment',\%changehash, - $udom,$uname); - if ($reqresult ne 'ok') { - push(@warn_coursereqs,$uname.':'.$udom); - } } } else { push(@processing_errors,$uname.':'.$udom); @@ -480,14 +1391,299 @@ sub update_request_queue { } else { push(@invalidusers,$uname.':'.$udom); } + } elsif ($context eq 'requestauthor') { + my ($num,$uname) = split(/:/,$item); + my $uhome = &Apache::lonnet::homeserver($uname,$cdom); + if ($uhome ne 'no_host') { + my ($user_is_adv,$user_is_author) = &Apache::lonnet::is_advanced_user($cdom,$uname); + if ($user_is_author) { + push(@existing,$uname); + } elsif (&Apache::lonnet::usertools_access($uname,$cdom,'requestauthor', + undef,'requestauthor')) { + if (&Apache::lonnet::allowed('cau',$cdom)) { + if (&Apache::lonnet::assignrole($cdom,$uname,'/'.$cdom.'/','au',undef,time,undef,undef,'requestauthor') eq 'ok') { + push(@completed,$uname); + &send_selfserve_notification($uname.':'.$cdom, + $approvedmsg,undef,undef,$now, + $beneficiary,$sender); + my %userrequest = ( + author => { + timestamp => $now, + adjudicator => $env{'user.name'}.':'.$env{'user.domain'}, + status => 'approved', + }, + author_status => 'approved', + ); + my $userresult = + &Apache::lonnet::put('requestauthor',\%userrequest,$cdom,$uname); + if ($userresult ne 'ok') { + push(@warn_approves,$uname.':'.$cdom); + } + } else { + push(@processing_errors,$uname); + } + } else { + push(@nopermissions,$uname); + } + } else { + push(@nopermissions,$uname); + } + } else { + push(@invalidusers,$uname.':'.$cdom); + } + push(@toremove,(@invalidusers,@nopermissions)); + } elsif ($context eq 'requestusername') { + my ($num,$uname) = split(/:/,$item); + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + my %curr = &Apache::lonnet::get($dbname,[$uname],$cdom,$domconfiguser); + + if (ref($curr{$uname}) eq 'HASH') { + my ($logtoken,$serverid,$encpass,$courseid,$id,$firstname, + $middlename,$lastname,$generation,$inststatus,$email); + $curr{$uname}{'timestamp'} = $now; + $curr{$uname}{'adjudicator'} = $env{'user.name'}.':'.$env{'user.domain'}; + $courseid = $curr{$uname}{'courseid'}; + $id = $curr{$uname}{'id'}; + $firstname = $curr{$uname}{'firstname'}; + $middlename = $curr{$uname}{'middlename'}; + $lastname = $curr{$uname}{'lastname'}; + $generation = $curr{$uname}{'generation'}; + $inststatus = $curr{$uname}{'inststatus'}; + + if ($curr{$uname}{'email'} ne '') { + $email = $curr{$uname}{'email'}; + } elsif ($uname =~ /^[^\@]+\@[^\@]+$/) { + $email = $uname; + } + + my $upass; + if ($curr{$uname}{'tmpinfo'}) { + my ($key,$caller)=split(/&/,$curr{$uname}{'tmpinfo'}); + if ($caller eq 'createaccount') { + if ($curr{$uname}{'upass'} eq '') { + $upass = $curr{$uname}{'upass'}; + } else { + $upass = &Apache::loncommon::des_decrypt($key,$curr{$uname}{'upass'}); + } + } else { + push(@processing_errors,$uname); + } + } else { + $upass = $curr{$uname}{'upass'}; + } + if ($upass eq '') { + push(@processing_errors,$uname); + } else { + undef($curr{$uname}{'upass'}); + my $result = + &Apache::lonnet::modifyuser($cdom,$uname,$id,'internal',$upass, + $firstname,$middlename,$lastname, + $generation,undef,undef,$email); + if ($result eq 'ok') { + $curr{$uname}{'status'} = 'created'; + push(@completed,$uname); + my $uhome = &Apache::lonnet::homeserver($uname,$cdom); + if ($uhome eq 'no_host') { + push(@warn_approves,$uname); + } else { + unless (($inststatus eq 'default') || ($inststatus eq '')) { + &Apache::lonnet::put('environment',{inststatus => $inststatus},$cdom,$uname); + } + &send_selfserve_notification($uname.':'.$cdom, + $approvedmsg,undef,undef,$now, + $beneficiary,$sender); + if (&Apache::lonnet::put($dbname,\%curr,$cdom,$domconfiguser) ne 'ok') { + push(@warn_approves,$uname); + } + } + } else { + push(@processing_errors,$uname); + } + } + } else { + push(@invalidusers,$uname); + } + push(@toremove,@invalidusers); + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my ($num,$extent,$role,$uname,$udom,$key,$logmsg,$result); + if ($context eq 'othdombydc') { + ($num,$extent,$role,$uname,$udom) = split(/:/,$item); + if ($udom eq $cdom) { + $key = 'pending:'.$uname.':'.$extent.':'.$role; + } + } elsif ($context eq 'othdombyuser') { + ($num,$extent,$role) = split(/:/,$item); + $key = 'pending:'.$extent.':'.$role; + $uname = $env{'user.name'}; + $udom = $env{'user.domain'}; + } + if (($key) && (exists($requesthash{$key}))) { + if (ref($requesthash{$key}) eq 'HASH') { + my $requester = $requesthash{$key}->{'requester'}; + my ($requname,$requdom) = split(/:/,$requester); + my $start = $requesthash{$key}->{'start'}; + my $end = $requesthash{$key}->{'end'}; + my $credits = $requesthash{$key}->{'credits'}; + my $reqcontext = $requesthash{$key}->{'context'}; + if ((($context eq 'othdombydc') && + (&Apache::lonnet::homeserver($uname,$udom) ne 'no_host')) || + ($context eq 'othdombyuser')) { + if (&Apache::lonnet::homeserver($requname,$requdom) ne 'no_host') { + if (($role eq 'ca') || ($role eq 'aa')) { + my ($audom,$auname) = ($extent =~ m{^/($match_domain)/($match_username)$}); + if (&Apache::lonnet::homeserver($auname,$audom) ne 'no_host') { + if ($requester eq $auname.':'.$audom) { + unless ($gotroles{$requester}) { + &requester_roles($auname,$audom,\%requesteractive); + $gotroles{$requester} = 1; + } + if (ref($requesteractive{$requester}) eq 'HASH') { + if ($requesteractive{$requester}{':'.$audom.':au'}) { + $result = &Apache::lonnet::assignrole($udom,$uname,$extent,$role, + $end,$start,'','',$reqcontext, + $context,$requester); + } + } + } + } + } elsif (($role eq 'co') || ($role eq 'cc')) { + my ($crsdom,$crsnum) = ($extent =~ m{^/($match_domain)/($match_courseid)}); + if (&Apache::lonnet::is_course($crsdom,$crsnum)) { + my %info = &Apache::lonnet::coursedescription("$crsdom/$crsnum",{'one_time' => 1}); + if ((($role eq 'co') && ($info{'type'} eq 'Community')) || + (($role eq 'cc') && ($info{'type'} ne 'Community'))) { + if ($info{'internal.courseowner'} eq $requester) { + unless ($gotroles{$requester}) { + &requester_roles($requname,$requdom,\%requesteractive); + $gotroles{$requester} = 1; + } + if (ref($requesteractive{$requester}) eq 'HASH') { + if ($requesteractive{$requester}{"$crsnum:$crsdom:$role"}) { + ($logmsg,$result) = + &Apache::loncommon::commit_standardrole($udom,$uname,$extent,$role,$start, + $end,$crsdom,$crsnum,'', + $reqcontext,'',$context, + $requester); + } + } + } + } + } + } elsif ($role =~ m{^(cr)/($match_domain)/($match_username)/(\w+)$}) { + my ($mrole,$crudom,$cruname,$rolename) = ($1,$2,$3,$4); + my ($crsdom,$crsnum,$csec) = + ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/(\w+))$}); + if (&Apache::lonnet::is_course($crsdom,$crsnum)) { + my ($rdummy,$roledef) = + &Apache::lonnet::get('roles',["rolesdef_$rolename"],$crudom,$cruname); + if (($rdummy ne 'con_lost') && ($roledef ne '')) { + unless ($gotroles{$requester}) { + &requester_roles($requname,$requdom,\%requesteractive); + $gotroles{$requester} = 1; + } + if (ref($requesteractive{$requester}) eq 'HASH') { + if (&requester_has_perm($crsdom,$crsnum,$mrole,$requesteractive{$requester})) { + ($logmsg,$result) = + &Apache::loncommon::commit_customrole($udom,$uname,$extent,$crudom,$cruname, + $rolename,$start,$end,$reqcontext, + $context,$requester); + } + } + } + } + } else { + my $process; + foreach my $type ('course','domain') { + if (grep(/^\Q$role\E$/,@{$roles_by_context{$type}})) { + if ($type eq 'course') { + my ($crsdom,$crsnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/(\w+))$}); + if (&Apache::lonnet::is_course($crsdom,$crsnum)) { + my $typeok; + if ($role eq 'cc') { + my %info = &Apache::lonnet::coursedescription("$crsdom/$crsnum",{'one_time' => 1}); + if ($info{'type'} eq 'Course') { + $typeok = 1; + } + } else { + $typeok = 1; + } + if ($typeok) { + unless ($gotroles{$requester}) { + &requester_roles($requname,$requdom,\%requesteractive); + $gotroles{$requester} = 1; + } + if (ref($requesteractive{$requester}) eq 'HASH') { + if (&requester_has_perm($crsdom,$crsnum,$role,$requesteractive{$requester})) { + ($logmsg,$result) = + &Apache::loncommon::commit_standardrole($udom,$uname,$extent,$role,$start, + $end,$crsdom,$crsnum,$csec, + $reqcontext,$credits,$context, + $requester); + } + } + } + } + } else { + my ($domain) = ($extent =~ m{^/($match_domain)/}); + if (&Apache::lonnet::domain($domain) ne '') { + unless ($gotroles{$requester}) { + &requester_roles($requname,$requdom,\%requesteractive); + $gotroles{$requester} = 1; + } + if (&requester_has_perm($domain,'',$role,$requesteractive{$requester})) { + $result = &Apache::lonnet::assignrole($udom,$uname,$extent,$role, + $end,$start,'','',$reqcontext, + $context,$requester); + } + } + } + last; + } + } + } + if ($result eq 'ok') { + $requestedby{$item} = $requester; + my $statusres; + my $id = $uname.':'.$udom.':'.$role; + if (($role eq 'ca') || ($role eq 'aa')) { + my ($audom,$auname) = ($extent =~ m{^/($match_domain)/($match_username)$}); + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'approved'},$audom,$auname); + } elsif ($extent =~ m{^/($match_domain)/$}) { + my $domain = $1; + my $configuser = &Apache::lonnet::get_domainconfiguser($domain); + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'approved'},$domain,$configuser); + } else { + my ($crsdom,$crsnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/([^/]+))$}); + $id .= ':'.$csec; + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'approved'},$crsdom,$crsnum); + } + if ($statusres eq 'ok') { + my $newkey; + if ($context eq 'othdombydc') { + $newkey = 'approved:'.$uname.':'.$extent.':'.$role; + } elsif ($context eq 'othdombyuser') { + $newkey = 'approved:'.$extent.':'.$role; + } + $requesthash{$newkey} = $requesthash{$key}; + delete($requesthash{$key}); + push(@toremove,$key); + push(@completed,$item); + } + } else { + push(@warn_approves,$key); + } + } + } + } + } } else { my ($num,$cnum) = split(':',$item); - if (ref($requesthash{$cnum.'_approval'}) eq 'HASH') { + if (ref($requesthash{$cnum.'_'.$queue}) eq 'HASH') { if (&Apache::lonnet::homeserver($cnum,$cdom) eq 'no_host') { - my $ownername = $requesthash{$cnum.'_approval'}{'ownername'}; - my $ownerdom = $requesthash{$cnum.'_approval'}{'ownerdom'}; - my $crstype = $requesthash{$cnum.'_approval'}{'crstype'}; - my $coursedesc = $requesthash{$cnum.'_approval'}{'description'}; + my $ownername = $requesthash{$cnum.'_'.$queue}{'ownername'}; + my $ownerdom = $requesthash{$cnum.'_'.$queue}{'ownerdom'}; + $crstype = $requesthash{$cnum.'_'.$queue}{'crstype'}; + my $coursedesc = $requesthash{$cnum.'_'.$queue}{'description'}; my $longroles = \%courseroles; if ($crstype eq 'community') { $longroles = \%communityroles; @@ -513,15 +1709,68 @@ sub update_request_queue { &Apache::lonnet::restore($requestkey,'courserequests', $ownerdom,$ownername); if ((ref($history{'details'}) eq 'HASH') && - ($history{'disposition'} eq 'approval')) { - my ($logmsg,$newusermsg,$addresult,$enrollcount,$response,$keysmsg); - my $result = &course_creation($cdom,$cnum,$context,$history{'details'},\$logmsg, - \$newusermsg,\$addresult,\$enrollcount, - \$response,\$keysmsg,\%domdefs,$longroles); + ($history{'disposition'} eq $queue)) { + my ($logmsg,$newusermsg,$addresult,$enrollcount,$response, + $keysmsg,$code,%customitems); + my $clonemsg = []; + my $fullname = ''; + my $inprocess = &Apache::lonnet::auto_crsreq_update($cdom,$cnum,$crstype,'process',$ownername, + $ownerdom,$fullname,$coursedesc); + if (ref($inprocess) eq 'HASH') { + foreach my $key (keys(%{$inprocess})) { + if (exists($history{'details'}{$key})) { + $customitems{$key} = $history{'details'}{$key}; + } + } + } + if ($history{'details'}{'clonecrs'}) { + $customitems{'_LC_clonefrom'} = $history{'details'}{'clonedom'}.'_'.$history{'details'}{'clonecrs'}; + } + my ($result,$postprocess) = &course_creation($cdom,$cnum,$context,$history{'details'}, + \$logmsg,$clonemsg,\$newusermsg,\$addresult,\$enrollcount, + \$response,\$keysmsg,\%domdefs,$longroles,\$code,\%customitems); if ($result eq 'created') { + if ($crstype eq 'community') { + $approvedmsg = $approvalmsg{'community'}; + } else { + $approvedmsg = $approvalmsg{'course'}; + } + my $firsturl = &Apache::lonnet::course_portal_url($cnum,$cdom); + if (ref($approvedmsg) eq 'ARRAY') { + if (ref($approvedmsg->[1]) eq 'HASH') { + $approvedmsg->[1]->{'args'} = [$firsturl]; + } + if ((ref($clonemsg) eq 'ARRAY') && (@{$clonemsg})) { + push(@{$approvedmsg},@{$clonemsg}); + } + if ($code) { + push(@{$approvedmsg}, + { + mt => 'Students can automatically select your course by entering this code: [_1]', + args => [$code], + }); + $codes{$cnum} = $code; + } + if (ref($postprocess) eq 'HASH') { + if (ref($postprocess->{'createdmsg'}) eq 'ARRAY') { + foreach my $item (@{$postprocess->{'createdmsg'}}) { + if (ref($item) eq 'HASH') { + if ($item->{'mt'} ne '') { + push(@{$approvedmsg},$item); + } + } + } + } + } + } push(@completed,$cnum); - &send_selfserve_notification($ownername.':'.$ownerdom,$approvedmsg, - $cid,$coursedesc,$now,$beneficiary,$sender); + + unless (&Apache::lonnet::del_dom($namespace,[$cnum.'_'.$queue],$cdom) eq 'ok') { + push(@warn_dels,$cnum); + } + &send_selfserve_notification($ownername.':'.$ownerdom, + $approvedmsg,$cid,$coursedesc,$now, + $beneficiary,$sender,undef,undef,$crstype); my %reqhash = ( reqtime => $history{'reqtime'}, crstype => $history{'crstype'}, @@ -567,43 +1816,130 @@ sub update_request_queue { } my @changes = (@completed,@rejections); if ($context eq 'domain') { + @changes = map {$_.'_'.$queue} (@changes); + } elsif ($context eq 'requestauthor') { @changes = map {$_.'_approval'} (@changes); + } elsif ($context eq 'requestusername') { + @changes = map {&escape($_).'_approval'} (@changes); } if (@rejections) { foreach my $item (@rejections) { - if ($context eq 'course') { - my $user = $item; + if (($context eq 'course') || ($context eq 'requestauthor')) { + my ($user,$uname,$udom,%userrequest,$key,$dbname); + if ($context eq 'requestauthor') { + $uname = $item; + $udom = $cdom; + $user = $uname.':'.$udom; + $key = 'author'; + $dbname = 'requestauthor'; + } else { + $user = $item; + ($uname,$udom) = split(/:/,$user); + $key = $cdom.'_'.$cnum; + $dbname = $namespace; + } &send_selfserve_notification($user,$rejectedmsg,$cid,$coursedesc, - $now,$beneficiary,$sender); - my ($uname,$udom) = split(/:/,$user); - my %userrequest = ( - $cdom.'_'.$cnum => { + $now,$beneficiary,$sender,undef,undef, + $crstype); + %userrequest = ( + $key => { timestamp => $now, adjudicator => $env{'user.name'}.':'.$env{'user.domain'}, - status => 'rejected', + status => 'rejection', } ); + if ($context eq 'requestauthor') { + $userrequest{'author_status'} = 'rejection'; + } + my $userresult = + &Apache::lonnet::put($dbname,\%userrequest,$udom,$uname); + if ($userresult ne 'ok') { + push(@warn_rejects,$item); + } + } elsif ($context eq 'requestusername') { + my ($uname,$udom,$dbname); + $uname = $item; + $udom = $cdom; + $dbname = 'nohist_requestedusernames'; + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + my %curr = &Apache::lonnet::get($dbname,[$uname],$cdom,$domconfiguser); + if (ref($curr{$uname}) eq 'HASH') { + $curr{$uname}{'status'} = 'rejected'; + $curr{$uname}{'timestamp'} = $now; + $curr{$uname}{'adjudicator'} = $env{'user.name'}.':'.$env{'user.domain'}; + undef($curr{$uname}{'tmpinfo'}); + undef($curr{$uname}{'upass'}); + } my $userresult = - &Apache::lonnet::put($namespace,\%userrequest,$udom,$uname); + &Apache::lonnet::put($dbname,\%curr,$cdom,$domconfiguser); if ($userresult ne 'ok') { - push(@warn_rejects,$user); + push(@warn_rejects,$uname); + } + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my ($extent,$role,$uname,$udom,$oldkey,$newkey); + my $dbname = 'nohist_othdomqueued'; + if ($context eq 'othdombydc') { + ($extent,$role,$uname,$udom) = split(/:/,$item); + $oldkey = 'pending:'.$uname.':'.$extent.':'.$role; + $newkey = 'rejected:'.$uname.':'.$extent.':'.$role; + } elsif ($context eq 'othdombyuser') { + ($extent,$role) = split(/:/,$item); + $oldkey = 'pending:'.$extent.':'.$role; + $newkey = 'rejected:'.$extent.':'.$role; + $uname = $env{'user.name'}; + $udom = $env{'user.domain'}; + } + if (exists($requesthash{$oldkey})) { + if (ref($requesthash{$oldkey}) eq 'HASH') { + my $statusres; + my $id = $uname.':'.$udom.':'.$role; + if (($role eq 'ca') || ($role eq 'aa')) { + my ($audom,$auname) = ($extent =~ m{^/($match_domain)/($match_username)$}); + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'rejected'},$audom,$auname); + } elsif ($extent =~ m{^/($match_domain)/\Q$role\E$}) { + my $domain = $1; + my $configuser = &Apache::lonnet::get_domainconfiguser($domain); + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'rejected'},$domain,$configuser); + } else { + my ($crsdom,$crsnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/([^/]+))$}); + $id .= ':'.$csec; + $statusres = &Apache::lonnet::put($dbname,{'status&'.$id => 'rejected'},$crsdom,$crsnum); + } + if ($statusres eq 'ok') { + $requesthash{$newkey} = $requesthash{$oldkey}; + delete($requesthash{$oldkey}); + push(@toremove,$oldkey); + $requesthash{$newkey}->{'timestamp'} = $now; + $requesthash{$newkey}->{'adjudicator'} = $env{'user.name'}.':'.$env{'user.domain'}; + $requestedby{$item} = $requesthash{$newkey}->{'requester'}; + push(@rejectedreqs,$item); + } else { + push(@warn_rejects,$oldkey); + } + } } } else { my $cnum = $item; - if (ref($requesthash{$cnum.'_approval'}) eq 'HASH') { + if (ref($requesthash{$cnum.'_'.$queue}) eq 'HASH') { if (&Apache::lonnet::homeserver($cnum,$cdom) eq 'no_host') { my $requestkey = $cdom.'_'.$cnum; - my $ownername = $requesthash{$cnum.'_approval'}{'ownername'}; - my $ownerdom = $requesthash{$cnum.'_approval'}{'ownerdom'}; - my $coursedesc = $requesthash{$cnum.'_approval'}{'description'}; + my $ownername = $requesthash{$cnum.'_'.$queue}{'ownername'}; + my $ownerdom = $requesthash{$cnum.'_'.$queue}{'ownerdom'}; + my $coursedesc = $requesthash{$cnum.'_'.$queue}{'description'}; + $crstype = $requesthash{$cnum.'_'.$queue}{'crstype'}; + if ($crstype eq 'community') { + $rejectedmsg = $rejectionmsg{'community'}; + } else { + $rejectedmsg = $rejectionmsg{'course'}; + } &send_selfserve_notification($ownername.':'.$ownerdom,$rejectedmsg, $cid,$coursedesc,$now,$beneficiary, - $sender); + $sender,undef,undef,$crstype); my %history = &Apache::lonnet::restore($requestkey,'courserequests', $ownerdom,$ownername); if ((ref($history{'details'}) eq 'HASH') && - ($history{'disposition'} eq 'approval')) { + ($history{'disposition'} eq $queue)) { my %reqhash = ( reqtime => $history{'reqtime'}, crstype => $history{'crstype'}, @@ -628,6 +1964,9 @@ sub update_request_queue { } else { push(@warn_rejects,$cnum); } + unless (&Apache::lonnet::del_dom($namespace,[$cnum.'_'.$queue],$cdom) eq 'ok') { + push(@warn_dels,$cnum); + } } else { push(@warn_rejects,$cnum); } @@ -640,10 +1979,56 @@ sub update_request_queue { } } } + if (@toremove) { + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($cdom); + foreach my $item (@toremove) { + if ($context eq 'requestauthor') { + my %userrequest = ( + author => { + timestamp => $now, + adjudicator => $env{'user.name'}.':'.$env{'user.domain'}, + status => 'deleted', + }, + author_status => 'deleted', + ); + &Apache::lonnet::put('requestauthor',\%userrequest,$cdom,$item); + } elsif ($context eq 'requestusername') { + my $dbname = 'nohist_requestedusernames'; + my %curr = &Apache::lonnet::get($dbname,[$item],$cdom,$domconfiguser); + if (ref($curr{$item}) eq 'HASH') { + $curr{$item}{'status'} = 'deleted'; + $curr{$item}{'timestamp'} = $now; + $curr{$item}{'adjudicator'} = $env{'user.name'}.':'.$env{'user.domain'}; + undef($curr{$item}{'upass'}); + undef($curr{$item}{'tmpinfo'}); + } + } + } + unless (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + @toremove = map {$_.'_approval'} (@toremove); + } + if (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my $delresult; + if ($context eq 'othdombyuser') { + $delresult = &Apache::lonnet::del($namespace,\@toremove,$env{'user.domain'},$env{'user.name'}); + } else { + $delresult = &Apache::lonnet::del_dom($namespace,\@toremove,$cdom); + } + unless ($delresult eq 'ok') { + push(@warn_dels,@toremove); + } + } + } if (@changes) { my $delresult; if ($context eq 'course') { $delresult = &Apache::lonnet::del($namespace,\@changes,$cdom,$cnum); + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + if ($context eq 'othdombydc') { + $delresult = &Apache::lonnet::put($namespace,\%requesthash,$cdom,$confname); + } elsif ($context eq 'othdombyuser') { + $delresult = &Apache::lonnet::put($namespace,\%requesthash,$env{'user.domain'},$env{'user.name'}); + } } else { $delresult = &Apache::lonnet::del_dom($namespace,\@changes,$cdom); } @@ -655,7 +2040,11 @@ sub update_request_queue { $chgmsg = "'Action was taken on the following enrollment requests by [_1].',$namelink"; if (@completed) { $approvedlist = join("\n",@completed); - $output .= '<p>'.&mt('The following were enrolled in the course:').'<ul>'; + if ($crstype eq 'community') { + $output .= '<p>'.&mt('The following were enrolled in the community:').'<ul>'; + } else { + $output .= '<p>'.&mt('The following were enrolled in the course:').'<ul>'; + } foreach my $user (@completed) { my ($uname,$udom) = split(/:/,$user); my $userlink = @@ -675,22 +2064,136 @@ sub update_request_queue { if ($notifylist ne '') { &send_selfserve_notification($notifylist,$chgmsg,$cid,$coursedesc, $now,'coursemanagers',$sender, + $approvedlist,$rejectedlist,$crstype); + } + } elsif ($context eq 'requestauthor') { + $chgmsg = "'Action was taken on the following Authoring Space requests by [_1].',$namelink"; + if (@completed) { + $approvedlist = join("\n",@completed); + $output .= '<p>'.&mt('The following requests were approved:').'<ul>'; + foreach my $uname (@completed) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; + } + if (@rejections) { + $rejectedlist = join("\n",@rejections); + $output .= '<p>'.&mt('The following requests were rejected:').'<ul>'; + foreach my $uname (@rejections) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; + } + if ($notifylist ne '') { + &send_selfserve_notification($notifylist,$chgmsg,undef,$domdesc, + $now,'authormanagers',$sender, $approvedlist,$rejectedlist); } + } elsif ($context eq 'requestusername') { + $chgmsg = "'Action was taken on the following LON-CAPA account requests by [_1].',$namelink"; + if (@completed) { + $approvedlist = join("\n",@completed); + $output .= '<p>'.&mt('The following requests were approved:').'<ul>'; + foreach my $uname (@completed) { + $output .= '<li>'.$uname.'</li>'; + + } + $output .= '</ul></p>'; + } + if (@rejections) { + $rejectedlist = join("\n",@rejections); + $output .= '<p>'.&mt('The following requests were rejected:').'<ul>'; + foreach my $uname (@rejections) { + $output .= '<li>'.$uname.'</li>'; + } + $output .= '</ul></p>'; + } + if ($notifylist ne '') { + &send_selfserve_notification($notifylist,$chgmsg,undef,$domdesc, + $now,'usernamemanagers',$sender, + $approvedlist,$rejectedlist); + } + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my @chgmsgs = ({ mt => 'Action was taken by [_1].', + args => [$namelink] }); + my (%approvals_by_requester,%rejections_by_requester,%for_requester); + my $sender = $env{'user.name'}.':'.$env{'user.domain'}; + if (@completed) { + my $msg; + if ($context eq 'othdombydc') { + $msg = &mt('The following roles in other domain(s) were assigned for user(s) in this domain:'); + } elsif ($context eq 'othdombyuser') { + $msg = &mt('The following roles in other domain(s) were assigned:'); + } + $output .= '<p>'.$msg.'<ul>'. + &get_othdomby_results($context,'approved',\@completed,\%approvals_by_requester,\%requestedby,\%for_requester). + '</ul></p>'; + } + if (@rejectedreqs) { + my $msg; + if ($context eq 'othdombydc') { + $msg = &mt('The following role assignments in other domain(s) for user(s) in this domain were rejected:'); + } elsif ($context eq 'othdombyuser') { + $msg = &mt('The following role assignments in other domain(s) were rejected:'); + } + $output .= '<p>'.$msg.'<ul>'. + &get_othdomby_results($context,'rejected',\@rejectedreqs,\%rejections_by_requester,\%requestedby,\%for_requester). + '</ul></p>'; + } + foreach my $key (sort(keys(%for_requester))) { + if (ref($approvals_by_requester{$key}) eq 'ARRAY') { + if (@{$approvals_by_requester{$key}} > 0) { + if ($context eq 'othdombydc') { + push(@chgmsgs,{ mt => '[_1]The following roles in other domain(s) were assigned:', + args => ["\n"]}); + } elsif ($context eq 'othdombyuser') { + push(@chgmsgs,{ mt => '[_1]The following roles in other domain(s) were accepted:', + args => ["\n"]}); + } + push(@chgmsgs,@{$approvals_by_requester{$key}}); + $approvedlist = $key; + } + } + if (ref($rejections_by_requester{$key}) eq 'ARRAY') { + if (@{$rejections_by_requester{$key}} > 0) { + if ($context eq 'othdombydc') { + push(@chgmsgs,{ mt => '[_1]The following roles in other domain(s) were rejected:', + args => ["\n"]}); + } elsif ($context eq 'othdombyuser') { + push(@chgmsgs,{ mt => '[_1]The following roles in other domain(s) were declined:', + args => ["\n"]}); + } + push(@chgmsgs,@{$rejections_by_requester{$key}}); + $rejectedlist = $key; + } + } + if (($approvedlist ne '') || ($rejectedlist ne '')) { + &send_selfserve_notification($key,\@chgmsgs,'',$domdesc,$now, + $context,$sender,$approvedlist, + $rejectedlist); + } + } } else { - $chgmsg = "'Action was taken on the following course requests by [_1].',$namelink"; + $chgmsg = "'Action was taken on the following course and community requests by [_1].',$namelink"; if (@completed) { $approvedlist = join("\n",@completed); - $output .= '<p>'.&mt('The following courses were created:').'<ul>'; + $output .= '<p>'.&mt('The following courses/communities were created:').'<ul>'; foreach my $cnum (@completed) { my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; } else { $showcourse = $cnum; } my $syllabuslink = &Apache::loncommon::syllabuswrapper($showcourse,$cnum,$cdom); + if ($codes{$cnum}) { + $syllabuslink .= ' '.&mt('Unique code: [_1]',$codes{$cnum}); + } $output .= '<li>'.$syllabuslink.'</li>'; } $output .= '</ul></p>'; @@ -700,8 +2203,8 @@ sub update_request_queue { $output .= '<p>'.&mt('The following requests were rejected:').'<ul>'; foreach my $cnum (@rejections) { my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; } else { $showcourse = $cnum; } @@ -712,9 +2215,14 @@ sub update_request_queue { if ($notifylist ne '') { &send_selfserve_notification($notifylist,$chgmsg,$cid,$domdesc, $now,'domainmanagers',$sender, - $approvedlist,$rejectedlist); + $approvedlist,$rejectedlist,$crstype); } } + } else { + if (($context eq 'requestauthor') || ($context eq 'requestusername') || + ($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + push(@warn_dels,@changes); + } } } if (@existing) { @@ -724,8 +2232,16 @@ sub update_request_queue { $output .= '<li>'.$user.'</li>'; } $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt('Authoring Space requests from the following users were deleted because one already exists:').'<ul>'; + foreach my $uname (@existing) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; } else { - $output .= '<p>'.&mt('The following course creation requests were deleted because the course has already been created:').'<ul>'; + $output .= '<p>'.&mt('The following course/community creation requests were deleted because the course or community has already been created:').'<ul>'; foreach my $cnum (@existing) { my $showcourse; my %coursehash = &Apache::lonnet::coursedescription($cdom.'/'.$cnum); @@ -746,13 +2262,20 @@ sub update_request_queue { $output .= '<li>'.$user.'</li>'; } $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt('The following requests were ignored because the request is no longer in the queue:').'<ul>'; + foreach my $uname (@missingreq) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; } else { - $output .= '<p>'.&mt('The following course creation requests were ignored because the request is no longer in the course request queue:').'<ul>'; + $output .= '<p>'.&mt('The following course/community creation requests were ignored because the request is no longer in the queue:').'<ul>'; foreach my $cnum (@missingreq) { $output .= '<li>'.$cnum.'</li>'; } $output .= '</ul></p>'; - } } if (@invalidusers) { @@ -762,6 +2285,14 @@ sub update_request_queue { $output .= '<li>'.$user.'</li>'; } $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt('The following Authoring Space requests were deleted because the requestor does not have a LON-CAPA account:').'<ul>'; + foreach my $uname (@invalidusers) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; } } if (@limitexceeded) { @@ -774,17 +2305,27 @@ sub update_request_queue { } } if (@nopermissions) { - $output .= '<p>'.&mt('The following course creation requests could not be processed because the course owner does hot have rights to create this type of course:').'<ul>'; - foreach my $cnum (@nopermissions) { - my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; - } else { - $showcourse = $cnum; + if ($context eq 'course') { + $output .= '<p>'.&mt('The following course/community creation requests could not be processed because the owner does not have rights to create this type of course:').'<ul>'; + foreach my $cnum (@nopermissions) { + my $showcourse; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; + } else { + $showcourse = $cnum; + } + $output .= '<li>'.$showcourse.'</li>'; } - $output .= '<li>'.$showcourse.'</li>'; + $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt('The following requests could not be processed because the requestor does not have rights to request an Authoring Space:').'<ul>'; + foreach my $uname (@nopermissions) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; } - $output .= '</ul></p>'; } if (@processing_errors) { if ($context eq 'course') { @@ -793,12 +2334,26 @@ sub update_request_queue { $output .= '<li>'.$user.'</li>'; } $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt('The following requests could not be processed because an error occurred:').'<ul>'; + foreach my $uname (@processing_errors) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; + } elsif ($context eq 'requestusername') { + $output .= '<p>'.&mt('The following requests could not be processed because an error occurred:').'<ul>'; + foreach my $uname (@processing_errors) { + $output .= '<li>'.$uname.'</li>'; + } + $output .= '</ul></p>'; } else { - $output .= '<p>'.&mt('The following course creation requests could not be processed because an error occurred:').'<ul>'; + $output .= '<p>'.&mt('The following course/community creation requests could not be processed because an error occurred:').'<ul>'; foreach my $cnum (@processing_errors) { my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; } else { $showcourse = $cnum; } @@ -808,11 +2363,11 @@ sub update_request_queue { } } if (@rejectionerrors) { - $output .= '<p>'.&mt('The following course creation request rejections could not be fully processed because an error occurred:').'<ul>'; + $output .= '<p>'.&mt('The following course/community creation request rejections could not be fully processed because an error occurred:').'<ul>'; foreach my $cnum (@rejectionerrors) { my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; } else { $showcourse = $cnum; } @@ -827,12 +2382,41 @@ sub update_request_queue { $output .= '<li>'.$user.'</li>'; } $output .= '</ul></p>'; + } elsif ($context eq 'requestauthor') { + $output .= '<p>'.&mt("For the following users, an error occurred when updating the user's own author request record:").'<ul>'; + foreach my $uname (@warn_approves,@warn_rejects) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; + } elsif ($context eq 'requestusername') { + $output .= '<p>'.&mt("For the following users, an error occurred when updating the account request record for the user:").'<ul>'; + foreach my $uname (@warn_approves,@warn_rejects) { + $output .= '<li>'.$uname.'</li>'; + } + $output .= '</ul></p>'; + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my $msg; + if ($context eq 'othdombydc') { + $msg = &mt("For the following assignments of roles in other domain(s) for users from this domain, an error occurred when updating status in the pending queue"); + } elsif ($context eq 'othdombyuser') { + $msg = &mt("For the following assignments of roles in other domain(s), an error occurred when updating status in the pending queue"); + } + $output .= '<p>'.$msg.'<ul>'; + if (@warn_approves) { + $output .= &get_othdomby_results($context,'dequeue_error',\@warn_approves); + } + if (@warn_rejects) { + $output .= &get_othdomby_results($context,'dequeue_error',\@warn_rejects); + } + $output .= '</ul></p>'; } else { - $output .= '<p>'.&mt("For the following course requests an error occurred when updating the requestor's own course requests record:").'<ul>'; + $output .= '<p>'.&mt("For the following course/community requests an error occurred when updating the requestor's own requests record:").'<ul>'; foreach my $cnum (@warn_approves,@warn_rejects) { my $showcourse; - if (ref($requesthash{$cnum.'_approval'})) { - $showcourse = $requesthash{$cnum.'_approval'}{'description'}; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; } else { $showcourse = $cnum; } @@ -841,16 +2425,208 @@ sub update_request_queue { $output .= '</ul></p>'; } } - if (@warn_coursereqs) { - $output .= '<p>'..&mt("For the following users, an error occurred when setting rights to request creation of Concept Test courses:").'<ul>'; - foreach my $user (@warn_coursereqs) { - $output .= '<li>'.$user.'</li>'; + if (@warn_dels) { + if ($context eq 'requestauthor') { + $output .= '<p>'.&mt("For the following requests an error occurred when removing the request from the queue:").'<ul>'; + foreach my $uname (@warn_dels) { + my $userlink = + &Apache::loncommon::aboutmewrapper(&Apache::loncommon::plainname($uname,$cdom),$uname,$cdom); + $output .= '<li>'.$userlink.'</li>'; + } + $output .= '</ul></p>'; + } elsif ($context eq 'requestusername') { + $output .= '<p>'.&mt("For the following requests an error occurred when removing the request from the queue:").'<ul>'; + foreach my $item (@warn_dels) { + my ($escuname) = split(/_/,$item); + $output .= '<li>'.&unescape($escuname).'</li>'; } $output .= '</ul></p>'; + } elsif (($context eq 'othdombydc') || ($context eq 'othdombyuser')) { + my $msg; + if ($context eq 'othdombydc') { + $msg = &mt("For the following queued role assignments an error occurred when removing the item from the queue:"); + } elsif ($context eq 'othdombyuser') { + $msg = &mt("For the following role assignments pending approval an error occurred when removing the item from the queue:"); + } + $output .= '<p>'.$msg. + '<ul>'.&get_othdomby_results($context,'dequeue_error',\@warn_dels).'</ul>'. + '</p>'; + } else { + $output .= '<p>'.&mt("For the following course/community requests an error occurred when removing requests from the pending queue:").'<ul>'; + foreach my $cnum (@warn_dels) { + my $showcourse; + if (ref($requesthash{$cnum.'_'.$queue})) { + $showcourse = $requesthash{$cnum.'_'.$queue}{'description'}; + } else { + $showcourse = $cnum; + } + $output .= '<li>'.$showcourse.'</li>'; + } + $output .= '</ul></p>'; + } } return $output; } +sub get_othdomby_results { + my ($context,$action,$items,$results,$requestedby,$for_requester) = @_; + return unless (ref($items) eq 'ARRAY'); + unless ($action eq 'dequeue_error') { + return unless ((ref($results) eq 'HASH') && (ref($requestedby) eq 'HASH') && + (ref($for_requester) eq 'HASH')); + } + my $output; + foreach my $item (@{$items}) { + my ($extent,$role,$uname,$udom); + if ($action eq 'approved') { + if ($context eq 'othdombydc') { + (my $num,$extent,$role,$uname,$udom) = split(/:/,$item); + } elsif ($context eq 'othdombyuser') { + (my $num,$extent,$role) = split(/:/,$item); + } + } elsif ($action eq 'rejected') { + if ($context eq 'othdombydc') { + ($extent,$role,$uname,$udom) = split(/:/,$item); + } elsif ($context eq 'othdombyuser') { + ($extent,$role) = split(/:/,$item); + } + } elsif ($action eq 'dequeue_error') { + if ($context eq 'othdombydc') { + if ($item =~ /^\d+:/) { + (my $num,$extent,$role,$uname,$udom) = split(/:/,$item); + } elsif ($item =~ /^pending:/) { + (my $oldstatus,$uname,$extent,$role) = split(/:/,$item); + $udom = $env{'request.role.domain'}; + } else { + ($extent,$role,$uname,$udom) = split(/:/,$item); + } + } elsif ($context eq 'othdombyuser') { + if ($item =~ /^\d+:/) { + (my $num,$extent,$role) = split(/:/,$item); + } elsif ($item =~ /^pending:/) { + (my $oldstatus,$extent,$role) = split(/:/,$item); + } else { + ($extent,$role) = split(/:/,$item); + } + } + } + if ($context eq 'othdombyuser') { + $uname = $env{'user.name'}; + $udom = $env{'user.domain'}; + } + my (@text,@msgs); + if ($context eq 'othdombydc') { + push(@text,&mt('User: [_1]',$uname)); + push(@msgs,{ mt => '[_1]User: [_2]', + args => ["\n",$uname]}); + } + if (($role eq 'ca') || ($role eq 'aa')) { + my $plainrole = &Apache::lonnet::plaintext($role); + my ($audom,$auname) = ($extent =~ m{^/($match_domain)/($match_username)$}); + my $title = &Apache::loncommon::plainname($auname,$audom); + my $domdesc = &Apache::lonnet::domain($audom); + push(@text,(&mt('Role: [_1]',$plainrole), + &mt('Domain: [_1]',$domdesc), + &mt('Authoring Space belonging to: [_1]',$title))); + push(@msgs,{ mt => 'Role: [_1]', + args => [$plainrole], + }, + { mt => 'Domain: [_1]', + args => [$domdesc], + }, + { mt => 'Authoring Space belonging to: [_1]', + args => ["$title\n"], + }); + } elsif ($extent =~ m{^/($match_domain)/$}) { + my $domain = $1; + my $domdesc = &Apache::lonnet::domain($domain); + my $plainrole = &Apache::lonnet::plaintext($role); + if ($domdesc ne '') { + push(@text,(&mt('Role: [_1]',$plainrole), + &mt('Domain: [_1]',$domdesc))); + push(@msgs,{ mt => 'Role: [_1]', + args => [$plainrole], + }, + { mt => 'Domain: [_1]', + args => ["$domdesc\n"], + }); + } + } else { + my ($crsdom,$crsnum,$csec) = ($extent =~ m{^/($match_domain)/($match_courseid)(?:|/([^/]+)$)}); + if (($crsdom ne '') && ($crsnum ne '')) { + my %info = &Apache::lonnet::coursedescription("$crsdom/$crsnum"); + my $plainrole = &Apache::lonnet::plaintext($role,$info{'type'}); + my $domdesc = &Apache::lonnet::domain($crsdom); + push(@text,(&mt('Role: [_1]',$plainrole), + &mt('Domain: [_1]',$domdesc), + &mt("$info{'type'}: [_1]",$info{'description'}))); + push(@msgs,{ mt => 'Role: [_1]', + args => [$plainrole], + }, + { mt => 'Domain: [_1]', + args => [$domdesc], + }); + if ($csec ne '') { + push(@text,&mt('Section: [_1]',$csec)); + push(@msgs,{ mt => "$info{'type'}: [_1]", + args => [$info{'description'}], + }, + { mt => 'Section: [_1]', + args => ["$info{'description'}\n"], + }); + } else { + push(@msgs,{ mt => "$info{'type'}: [_1]", + args => ["$info{'description'}\n"], + }); + } + } + } + $output .= '<li><ul><li>'.join('</li><li>',@text).'</li></ul></li>'; + unless ($action eq 'dequeue_error') { + push(@{$results->{$requestedby->{$item}}},@msgs); + $for_requester->{$requestedby->{$item}} = 1; + } + } + return $output; +} + +sub requester_roles { + my ($requname,$requdom,$activeroles) = @_; + if (ref($activeroles) eq 'HASH') { + my %roleshash = &Apache::lonnet::get_my_roles($requname,$requdom,'userroles'); + $activeroles->{$requname.':'.$requdom} = \%roleshash; + } + return; +} + +sub requester_has_perm { + my ($crsdom,$crsnum,$mrole,$requesterroles) = @_; + return unless (ref($requesterroles) eq 'HASH'); + my $has_perm; + foreach my $key (keys(%{$requesterroles})) { + if ($crsnum eq '') { + next unless ($key =~ /^\Q:$crsdom:\E/); + } else { + next unless (($key =~ /^\Q$crsnum:$crsdom:\E/) || ($key =~ /^\Q:$crsdom:\E/)); + } + my ($keycrs,$keydom,$keyrole) = split(/:/,$key); + if (($keycrs ne '') && ($crsnum ne '')) { + if ($keycrs eq $crsnum) { + if ($Apache::lonnet::pr{$keyrole.':c'} =~ /(^|:)c\Q$mrole\E(&|:)/) { + $has_perm = 1; + last; + } + } + } else { + if ($Apache::lonnet::pr{$keyrole.':d'} =~ /(^|:)c\Q$mrole\E(&|:)/) { + $has_perm = 1; + last; + } + } + } + return $has_perm; +} + sub get_student_counts { my ($cdom,$cnum) = @_; my (%idx,%stucounts); @@ -870,14 +2646,29 @@ sub get_student_counts { } sub course_creation { - my ($dom,$cnum,$context,$details,$logmsg,$newusermsg,$addresult,$enrollcount,$output, - $keysmsg,$domdefs,$longroles) = @_; + my ($dom,$cnum,$context,$details,$logmsg,$clonemsg,$newusermsg,$addresult, + $enrollcount,$output,$keysmsg,$domdefs,$longroles,$coderef,$customhash, + $callercontext,$user_lh) = @_; unless ((ref($details) eq 'HASH') && (ref($domdefs) eq 'HASH') && (ref($longroles) eq 'HASH')) { - return 'error: Invalid request'; + return ('error: Invalid request'); } - my ($result,$ownername,$ownerdom); + my ($result,$ownername,$ownerdom,$autocoowner); my $crstype = $details->{'crstype'}; + my $coursedesc = $details->{'cdescr'}; + my $accessstart = $details->{'accessstart'}; + my $accessend = $details->{'accessend'}; + my %domconfig = &Apache::lonnet::get_dom('configuration',['requestcourses','autoenroll'],$dom); + if (ref($domconfig{'requestcourses'}) eq 'HASH') { + if (ref($domconfig{'requestcourses'}{'uniquecode'}) eq 'HASH') { + if ($domconfig{'requestcourses'}{'uniquecode'}{$crstype}) { + $details->{'uniquecode'} = 1; + } + } + } + if (ref($domconfig{'autoenroll'}) eq 'HASH') { + $autocoowner = $domconfig{'autoenroll'}{'co-owners'}; + } if ($context eq 'domain') { $ownername = $details->{'owner'}; $ownerdom = $details->{'domain'}; @@ -885,40 +2676,124 @@ sub course_creation { $ownername = $env{'user.name'}; $ownerdom = $env{'user.domain'}; } - my $type = 'Course'; - if ($crstype eq 'community') { - $type = 'Community'; - } + my $fullname = &Apache::loncommon::plainname($ownername,$ownerdom); my $owneremail; my %emails = &Apache::loncommon::getemails($ownername,$ownerdom); foreach my $email ('permanentemail','critnotification','notification') { $owneremail = $emails{$email}; last if ($owneremail ne ''); } - my %reqdetails = &build_batchcreatehash($dom,$context,$details,$owneremail,$domdefs); + my %reqdetails = &build_batchcreatehash($dom,$cnum,$context,$details,$owneremail,$domdefs); my $cid = &LONCAPA::batchcreatecourse::build_course($dom,$cnum,'requestcourses', - \%reqdetails,$longroles,\$logmsg,\$newusermsg,\$addresult, - \$enrollcount,\$output,\$keysmsg,$ownerdom,$ownername,$cnum,$crstype); + \%reqdetails,$longroles,$logmsg,$clonemsg,$newusermsg,$addresult, + $enrollcount,$output,$keysmsg,$ownerdom,$ownername,$cnum,$crstype, + $coderef,$callercontext,$user_lh); + my $postprocess; if ($cid eq "/$dom/$cnum") { $result = 'created'; + my $code; + if (ref($coderef)) { + $code = $$coderef; + } + $postprocess = &Apache::lonnet::auto_crsreq_update($dom,$cnum,$crstype,$result,$ownername, + $ownerdom,$fullname,$coursedesc,$code, + $accessstart,$accessend,$customhash); + if ($autocoowner) { + my $instcode = $details->{'instcode'}; + if (($instcode ne '') && (ref($reqdetails{'users'}) eq 'HASH')) { + my @posscoowners; + my $now = time; + foreach my $person (keys(%{$reqdetails{'users'}})) { + my ($uname,$udom) = split(/:/,$person); + next if (($udom ne $dom) || (($uname eq $ownername) && ($udom eq $ownerdom))); + if ((&Apache::lonnet::homeserver($uname,$udom,1) ne 'no_host') && + (ref($reqdetails{'users'}{$person}) eq 'HASH')) { + if ((grep(/^cc$/,keys(%{$reqdetails{'users'}{$person}}))) && + (ref($reqdetails{'users'}{$person}{'cc'}) eq 'HASH')) { + my $start = $reqdetails{'users'}{$person}{'cc'}{'start'}; + my $end = $reqdetails{'users'}{$person}{'cc'}{'end'}; + if ((($start eq '') || ($start <= $now)) && + (($end eq '') || ($end >= $now))) { + push(@posscoowners,$person); + } + } + } + } + my @coowners; + if (@posscoowners) { + foreach my $user (@posscoowners) { + my ($checkcc,$desc) = + &Apache::lonnet::auto_validate_instcode($cnum,$dom,$instcode,$user); + unless ($checkcc eq 'valid') { + if (ref($reqdetails{'crosslists'}) eq 'HASH') { + foreach my $key (keys(%{$reqdetails{'crosslists'}})) { + if (ref($reqdetails{'crosslists'}{$key}) eq 'HASH') { + my $inst_crosslist = $reqdetails{'crosslists'}{$key}{'inst'}; + if ($inst_crosslist ne '') { + $checkcc = + &Apache::lonnet::auto_validate_inst_crosslist($cnum,$dom,$instcode, + $inst_crosslist,$user); + last if ($checkcc eq 'valid'); + } + } + } + } + } + if ($checkcc eq 'valid') { + if (@coowners > 0) { + unless (grep(/^\Q$user\E$/,@coowners)) { + push(@coowners,$user); + } + } else { + push(@coowners,$user); + } + } + } + } + if (@coowners > 0) { + my $chome = &Apache::lonnet::homeserver($cnum,$dom); + unless ($chome eq 'no_host') { + &Apache::lonnet::store_coowners($dom,$cnum,$chome,'',@coowners); + } + } + } + } } else { $result = 'error: '.$cid; } - return $result; + return ($result,$postprocess); } sub build_batchcreatehash { - my ($dom,$context,$details,$owneremail,$domdefs) = @_; + my ($dom,$cnum,$context,$details,$owneremail,$domdefs) = @_; my %batchhash; - my @items = qw{owner domain coursehome clonecrs clonedom datemode dateshift enrollstart enrollend accessstart accessend sections crosslists users}; + my @items = qw{owner domain coursehome clonecrs clonedom datemode dateshift tinyurls enrollstart enrollend accessstart accessend sections users uniquecode}; if ((ref($details) eq 'HASH') && (ref($domdefs) eq 'HASH')) { - my $emailenc = &Apache::lonnet::escape($owneremail); + my $emailenc = &escape($owneremail); my $owner = $details->{'owner'}.':'.$details->{'domain'}; foreach my $item (@items) { $batchhash{$item} = $details->{$item}; } + if (ref($details->{'crosslists'}) eq 'HASH') { + foreach my $key (keys(%{$details->{'crosslists'}})) { + if (ref($details->{'crosslists'}->{$key}) eq 'HASH') { + my $instsec = $details->{crosslists}->{$key}->{instsec}; + $batchhash{'crosslists'}{$key}{'inst'} = $details->{crosslists}->{$key}->{instcode}; + my $crskey = $cnum.':'.$batchhash{'crosslists'}{$key}{'inst'}; + my %formatted = &Apache::lonnet::auto_instsec_reformat($dom,'clutter', + {$crskey => [$instsec]}); + if (ref($formatted{$crskey}) eq 'ARRAY') { + $batchhash{'crosslists'}{$key}{'inst'} .= $formatted{$crskey}->[0]; + } + $batchhash{'crosslists'}{$key}{'loncapa'} = $details->{crosslists}->{$key}->{loncapa}; + } + } + } $batchhash{'title'} = $details->{'cdescr'}; $batchhash{'coursecode'} = $details->{'instcode'}; + if ($domdefs->{'officialcredits'} || $domdefs->{'unofficialcredits'}) { + $batchhash{'defaultcredits'} = $details->{'defaultcredits'}; + } $batchhash{'emailenc'} = $emailenc; $batchhash{'adds'} = $details->{'autoadds'}; $batchhash{'drops'} = $details->{'autodrops'}; @@ -926,7 +2801,17 @@ sub build_batchcreatehash { $batchhash{'authparam'} = $domdefs->{'auth_arg_def'}; if ($details->{'crstype'} eq 'community') { $batchhash{'crstype'} = 'Community'; + } elsif ($details->{'crstype'} eq 'placement') { + $batchhash{'crstype'} = 'Placement'; } else { + if ($details->{'crstype'} eq 'textbook') { + if ($details->{'clonecrs'} && $details->{'clonedom'}) { + my %clonedfrom = &Apache::lonnet::coursedescription($details->{'clonedom'}.'_'.$details->{'clonecrs'}); + $batchhash{'textbook'} = $clonedfrom{'description'}; + } + } elsif ($details->{'crstype'} eq 'lti') { + $batchhash{'lti'} = 1; + } $batchhash{'crstype'} = 'Course'; } my ($owner_firstname,$owner_lastname); @@ -965,29 +2850,90 @@ sub build_batchcreatehash { $batchhash{'users'}{$owner}{lastname} = $owner_lastname; $batchhash{'users'}{$owner}{emailenc} = $emailenc; $batchhash{'users'}{$owner}{owneremail} = $owneremail; + $batchhash{'setcomment'} = 1; } return %batchhash; } sub can_clone_course { - my ($uname,$udom,$clonecrs,$clonedom) = @_; + my ($uname,$udom,$clonecrs,$clonedom,$crstype,$dom,$instcode) = @_; my $canclone; + my $ccrole = 'cc'; + if ($crstype eq 'community') { + $ccrole = 'co'; + } my %roleshash = &Apache::lonnet::get_my_roles($uname,$udom,'userroles',['active'], - ['cc'],[$clonedom]); - if (exists($roleshash{$clonecrs.':'.$clonedom.':cc'})) { + [$ccrole],[$clonedom]); + if (exists($roleshash{$clonecrs.':'.$clonedom.':'.$ccrole})) { $canclone = 1; } else { - my %courseenv = &Apache::lonnet::userenvironment($clonedom,$clonecrs,('cloners')); + my %courseenv = &Apache::lonnet::userenvironment($clonedom,$clonecrs, + ('cloners','internal.coursecode')); my $cloners = $courseenv{'cloners'}; + my $clonefromcode = $courseenv{'internal.coursecode'}; if ($cloners ne '') { my @cloneable = split(',',$cloners); if (grep(/^\*$/,@cloneable)) { $canclone = 1; - } - if (grep(/^\*:\Q$udom\E$/,@cloneable)) { + } elsif (grep(/^\*:\Q$udom\E$/,@cloneable)) { + $canclone = 1; + } elsif (grep(/^\Q$uname\E:\Q$udom\E$/,@cloneable)) { $canclone = 1; } - if (grep(/^\Q$uname\E:\Q$udom\E$/,@cloneable)) { + unless ($canclone) { + if (($clonefromcode) && ($instcode) && ($clonedom eq $dom)) { + my (%gotdomdefaults,%gotcodedefaults); + foreach my $cloner (@cloneable) { + if (($cloner ne '*') && ($cloner !~ /^\*\:$match_domain$/) && + ($cloner !~ /^$match_username\:$match_domain$/) && ($cloner ne '')) { + if ($cloner =~ /\=/) { + my (%codedefaults,@code_order); + if (ref($gotcodedefaults{$clonedom}) eq 'HASH') { + if (ref($gotcodedefaults{$clonedom}{'defaults'}) eq 'HASH') { + %codedefaults = %{$gotcodedefaults{$clonedom}{'defaults'}}; + } + if (ref($gotcodedefaults{$clonedom}{'order'}) eq 'ARRAY') { + @code_order = @{$gotcodedefaults{$dom}{'order'}}; + } + } else { + &Apache::lonnet::auto_instcode_defaults($clonedom, + \%codedefaults, + \@code_order); + $gotcodedefaults{$clonedom}{'defaults'} = \%codedefaults; + $gotcodedefaults{$clonedom}{'order'} = \@code_order; + } + if (@code_order > 0) { + if (&Apache::lonnet::check_instcode_cloning(\%codedefaults,\@code_order, + $cloner,$clonefromcode,$instcode)) { + $canclone = 1; + last; + } + } + } + } + } + } + } + } else { + my %domdefs = &Apache::lonnet::get_domain_defaults($clonedom); + if ($domdefs{'canclone'}) { + unless ($domdefs{'canclone'} eq 'none') { + if ($domdefs{'canclone'} eq 'domain') { + if ($udom eq $clonedom) { + $canclone = 1; + } + } elsif (($clonefromcode) && ($instcode) && + ($clonedom eq $dom)) { + if (&Apache::lonnet::default_instcode_cloning($clonedom,$domdefs{'canclone'}, + $clonefromcode,$instcode)) { + $canclone = 1; + } + } + } + } + } + unless ($canclone) { + if (&Apache::lonnet::is_course_owner($clonedom,$clonecrs,$uname,$udom)) { $canclone = 1; } } @@ -995,4 +2941,597 @@ sub can_clone_course { return $canclone; } +sub get_processtype { + my ($context,$uname,$udom,$isadv,$dom,$crstype,$inststatuses,$domconfig) = @_; + return unless ((ref($inststatuses) eq 'ARRAY') && (ref($domconfig) eq 'HASH')); + if ($uname eq '' || $udom eq '') { + $uname = $env{'user.name'}; + $udom = $env{'user.domain'}; + $isadv = $env{'user.adv'}; + } + my (%userenv,%settings,$val,@options,$envkey); + if ($context eq 'course') { + @options = ('autolimit','validate','approval'); + $envkey = 'requestcourses.'.$crstype; + if (ref($domconfig->{'requestcourses'}) eq 'HASH') { + if (ref($domconfig->{'requestcourses'}->{$crstype}) eq 'HASH') { + %settings = %{$domconfig->{'requestcourses'}->{$crstype}}; + } + } + } else { + @options = ('automatic','approval'); + $envkey = 'requestauthor'; + if (ref($domconfig->{'requestauthor'}) eq 'HASH') { + %settings = %{$domconfig->{'requestauthor'}}; + } + } + if (($dom eq $udom) || ($context eq 'requestauthor')) { + %userenv = + &Apache::lonnet::userenvironment($udom,$uname,$envkey,'inststatus'); + if ($userenv{$envkey}) { + $val = $userenv{$envkey}; + @{$inststatuses} = ('_custom_'); + } else { + my %alltasks; + if (($isadv) && ($settings{'_LC_adv'} ne '')) { + $val = $settings{'_LC_adv'}; + @{$inststatuses} = ('_LC_adv_'); + } else { + if ($userenv{'inststatus'} ne '') { + @{$inststatuses} = split(',',$userenv{'inststatus'}); + } else { + @{$inststatuses} = ('default'); + } + foreach my $status (@{$inststatuses}) { + if (exists($settings{$status})) { + my $value = $settings{$status}; + next unless ($value); + unless (exists($alltasks{$value})) { + if (ref($alltasks{$value}) eq 'ARRAY') { + unless(grep(/^\Q$status\E$/,@{$alltasks{$value}})) { + push(@{$alltasks{$value}},$status); + } + } else { + @{$alltasks{$value}} = ($status); + } + } + } + } + my $maxlimit = 0; + if ($context eq 'course') { + foreach my $key (sort(keys(%alltasks))) { + if ($key =~ /^autolimit=(\d*)$/) { + if ($1 eq '') { + $val ='autolimit='; + last; + } elsif ($1 > $maxlimit) { + $maxlimit = $1; + } + } + } + } + if (($context eq 'requestauthor') || (!$val)) { + if ($context eq 'course' && $maxlimit) { + $val = 'autolimit='.$maxlimit; + } else { + foreach my $option (@options) { + if ($alltasks{$option}) { + $val = $option; + last; + } + } + } + } + } + } + } else { + %userenv = &Apache::lonnet::userenvironment($udom,$uname,'reqcrsotherdom.'.$crstype); + if ($userenv{'reqcrsotherdom.'.$crstype}) { + my @doms = split(',',$userenv{'reqcrsotherdom.'.$crstype}); + my $optregex = join('|',@options); + foreach my $item (@doms) { + my ($extdom,$extopt) = split(':',$item); + if ($extdom eq $dom) { + if ($extopt =~ /^($optregex)(=?\d*)$/) { + $val = $1.$2; + } + last; + } + } + @{$inststatuses} = ('_external_'); + } + } + return $val; +} + +sub queued_selfenrollment { + my ($notitle) = @_; + my $output; + my %selfenrollrequests = &Apache::lonnet::dump('selfenrollrequests'); + my %reqs_by_date; + foreach my $item (keys(%selfenrollrequests)) { + if (ref($selfenrollrequests{$item}) eq 'HASH') { + if ($selfenrollrequests{$item}{'status'} eq 'request') { + if ($selfenrollrequests{$item}{'timestamp'}) { + push(@{$reqs_by_date{$selfenrollrequests{$item}{'timestamp'}}},$item); + } + } + } + } + if (keys(%reqs_by_date)) { + unless ($notitle) { + $output .= '<br /><b>'.&mt('Enrollment requests pending Course Coordinator approval').'</b><br />'; + } + $output .= &Apache::loncommon::start_data_table(). + &Apache::loncommon::start_data_table_header_row(). + '<th>'.&mt('Date requested').'</th><th>'.&mt('Course title').'</th>'. + '<th>'.&mt('User role').'</th><th>'.&mt('Section').'</th>'. + &Apache::loncommon::end_data_table_header_row(); + my @sorted = sort { $a <=> $b } (keys(%reqs_by_date)); + foreach my $item (@sorted) { + if (ref($reqs_by_date{$item}) eq 'ARRAY') { + foreach my $crs (@{$reqs_by_date{$item}}) { + my %courseinfo = &Apache::lonnet::coursedescription($crs); + my $usec = $selfenrollrequests{$crs}{'section'}; + my $rolename = &Apache::lonnet::plaintext('st',$courseinfo{'type'},$crs); + if ($usec eq '') { + $usec = &mt('No section'); + } + $output .= &Apache::loncommon::start_data_table_row(). + '<td>'.&Apache::lonlocal::locallocaltime($item).'</td>'. + '<td>'.$courseinfo{'description'}.'</td>'. + '<td>'.$rolename.'</td><td>'.$usec.'</td>'. + &Apache::loncommon::end_data_table_row(); + } + } + } + $output .= &Apache::loncommon::end_data_table(); + } + return $output; +} + +sub update_coursereq_status { + my ($reqhash,$dom,$cnum,$reqstatus,$context,$udom,$uname) = @_; + my ($storeresult,$statusresult,$output); + my $requestkey = $dom.'_'.$cnum; + if ($requestkey =~ /^($match_domain)_($match_courseid)$/) { + $storeresult = &Apache::lonnet::store_userdata($reqhash,$requestkey, + 'courserequests',$udom,$uname); + if ($storeresult eq 'ok') { + my %status = ( + 'status:'.$dom.':'.$cnum => $reqstatus, + ); + $statusresult = &Apache::lonnet::put('courserequests',\%status,$udom,$uname); + } + } else { + $storeresult = 'error: invalid requestkey format'; + } + if ($storeresult ne 'ok') { + $output = &mt('An error occurred saving a record of the details of your request: [_1].',$storeresult); + if ($context eq 'domain') { + $output .= "\n"; + } else { + $output = '<span class="LC_warning">'.$output.'</span><br />'; + } + &Apache::lonnet::logthis("Error saving course request - $requestkey for $uname:$udom - $storeresult"); + } elsif ($statusresult ne 'ok') { + $output = &mt('An error occurred saving a record of the status of your request: [_1].',$statusresult); + if ($context eq 'domain') { + $output .= "\n"; + } else { + $output = '<span class="LC_warning">'.$output.'</span><br />'; + } + &Apache::lonnet::logthis("Error saving course request status for $requestkey (for $uname:$udom) - $statusresult"); + } + return ($storeresult,$output); +} + +sub process_official_reqs { + my ($context,$dom,$dcname,$dcdom) = @_; + my $reqsnamespace = 'courserequestqueue'; + my %requesthash = + &Apache::lonnet::dump_dom($reqsnamespace,$dom,'_pending'); + my (%newcids,%longroles,%stillpending); + my @courseroles = ('cc','in','ta','ep','ad','st'); + foreach my $role (@courseroles) { + $longroles{$role}=&Apache::lonnet::plaintext($role); + } + my %domdefs = &Apache::lonnet::get_domain_defaults($dom); + my ($output,$linefeed,$user_lh); + if ($context eq 'auto') { + $linefeed = "\n"; + $user_lh = &Apache::loncommon::user_lang($dcname,$dcdom); + } else { + $linefeed = '<br />'."\n"; + } + foreach my $key (keys(%requesthash)) { + my ($cnum,$status) = split('_',$key); + next if (&Apache::lonnet::homeserver($cnum,$dom) ne 'no_host'); + if (ref($requesthash{$key}) eq 'HASH') { + my $ownername = $requesthash{$key}{'ownername'}; + my $ownerdom = $requesthash{$key}{'ownerdom'}; + next if (&Apache::lonnet::homeserver($ownername,$ownerdom) eq 'no_host'); + my $inststatus; + my %userenv = + &Apache::lonnet::get('environment',['inststatus'], + $ownerdom,$ownername); + my ($tmp) = keys(%userenv); + if ($tmp !~ /^(con_lost|error|no_such_host)/i) { + $inststatus = $userenv{'inststatus'}; + } else { + undef(%userenv); + } + my $reqkey = $dom.'_'.$cnum; + my %history = &Apache::lonnet::restore($reqkey,'courserequests', + $ownerdom,$ownername); + if (ref($history{'details'}) eq 'HASH') { + my $instcode = $history{'details'}{'instcode'}; + my $crstype = $history{'details'}{'crstype'}; + my $reqtime = $history{'details'}{'reqtime'}; + my $cdescr = $history{'details'}{'cdescr'}; + my @currsec; + my $sections = $history{'details'}{'sections'}; + if (ref($sections) eq 'HASH') { + foreach my $i (sort(keys(%{$sections}))) { + if (ref($sections->{$i}) eq 'HASH') { + my $sec = $sections->{$i}{'inst'}; + if (!grep(/^\Q$sec\E$/,@currsec)) { + push(@currsec,$sec); + } + } + } + } + my $instseclist = join(',',@currsec); + my ($validationchk,$disposition,$reqstatus,$message, + $validation,$validationerror); + $validationchk = + &Apache::lonnet::auto_courserequest_validation($dom, + $ownername.':'.$ownerdom,$crstype,$inststatus, + $instcode,$instseclist); + if ($validationchk =~ /:/) { + ($validation,$message) = split(':',$validationchk); + } else { + $validation = $validationchk; + } + if ($validation =~ /^error(.*)$/) { + $disposition = 'approval'; + $validationerror = $1; + } else { + $disposition = $validation; + } + $reqstatus = $disposition; + if ($disposition eq 'process') { + my ($logmsg,$newusermsg,$addresult,$enrollcount,$response,$keysmsg,$code); + my $clonemsg = []; + my %customitems; + my $fullname = &Apache::loncommon::plainname($ownername,$ownerdom); + my $inprocess = &Apache::lonnet::auto_crsreq_update($dom,$cnum,$crstype,'process',$ownername, + $ownerdom,$fullname,$cdescr); + if (ref($inprocess) eq 'HASH') { + foreach my $key (keys(%{$inprocess})) { + if (exists($history{'details'}{$key})) { + $customitems{$key} = $history{'details'}{$key}; + } + } + } + if ($history{'details'}{'clonecrs'}) { + $customitems{'_LC_clonefrom'} = $history{'details'}{'clonedom'}.'_'.$history{'details'}{'clonecrs'}; + } + my ($result,$postprocess) = + &course_creation($dom,$cnum,'domain',$history{'details'},\$logmsg,$clonemsg,\$newusermsg, + \$addresult,\$enrollcount,\$response,\$keysmsg,\%domdefs,\%longroles, + \$code,\%customitems,$context,$user_lh); + if ($result eq 'created') { + $disposition = 'created'; + $reqstatus = 'created'; + my $cid = $dom.'_'.$cnum; + push(@{$newcids{$instcode}},$cid); + if ($dcname && $dcdom) { + my $firsturl = &Apache::lonnet::course_portal_url($cnum,$dom); + my $beneficiary = 'pendingrequestor'; + my $now = time; + my $owner = $ownername.':'.$ownerdom; + my $approvedmsg = + [{ + mt => 'Your requested course: [_1], (queued pending validation) has now been created.', + args => [$cdescr], + }]; + if ((ref($clonemsg) eq 'ARRAY') && (@{$clonemsg})) { + push(@{$approvedmsg},@{$clonemsg}); + } + push(@{$approvedmsg}, + { + mt => 'Visit [_1] to log-in and access the course.', + args => [$firsturl], + }, + { + mt => 'If currently logged-in to LON-CAPA, log-out and log-in again to select your new course role.', + args => [], + } + ); + my $sender = $dcname.':'.$dcdom; + if (ref($postprocess) eq 'HASH') { + if (ref($postprocess->{'createdmsg'}) eq 'ARRAY') { + foreach my $item (@{$postprocess->{'createdmsg'}}) { + if (ref($item) eq 'HASH') { + if ($item->{'mt'} ne '') { + push(@{$approvedmsg},$item); + } + } + } + } + if (ref($postprocess->{'createdactions'}) eq 'HASH') { + if (ref($postprocess->{'createdactions'}{'environment'}) eq 'HASH') { + &postprocess_crsenv($dom,$cnum,$postprocess->{'createdactions'}{'environment'}); + } + } + } + &send_selfserve_notification($owner,$approvedmsg, + $cid,$cdescr,$now, + $beneficiary,$sender, + undef,undef,$crstype); + } + } + } elsif ($disposition eq 'rejected') { + $output .= &mt('Queued course request for [_1] submitted by [_2] with status [_3] rejected when validating.',$instcode,$ownername.':'.$ownerdom,$inststatus).$linefeed; + } elsif ($disposition eq 'approval') { + $output .= &mt('Queued course request for [_1] submitted by [_2] with status [_3] switched to "approval by DC" because of validation error: [_4].',$instcode,$ownername.':'.$ownerdom,$inststatus,$validationerror).$linefeed; + + my $requestid = $cnum.'_'.$disposition; + my $request = { + $requestid => { + timestamp => $reqtime, + crstype => $crstype, + ownername => $ownername, + ownerdom => $ownerdom, + description => $cdescr, + }, + }; + my $putresult = &Apache::lonnet::newput_dom('courserequestqueue',$request,$dom); + unless ($putresult eq 'ok') { + $output .= &mt("An error occurred saving the modified course request for [_1] submitted by [_2] in the domain's courserequestqueue.db.",$instcode,$ownername.':'.$ownerdom).$linefeed; + } + } elsif ($disposition eq 'pending') { + my $instcode = $requesthash{$key}{'instcode'}; + my $description = $requesthash{$key}{'description'}; + my $timestamp = $requesthash{$key}{'timestamp'}; + my $entry = $cnum.':'.$ownername.':'.$ownerdom.':'. + $instcode.':'.$description; + if (ref($stillpending{$timestamp}) eq 'ARRAY') { + push(@{$stillpending{$timestamp}},$entry); + } else { + $stillpending{$timestamp} = [$entry]; + } + } + unless ($disposition eq 'pending') { + my ($statusresult,$output) = + &update_coursereq_status(\%requesthash,$dom,$cnum, + $reqstatus,'domain',$ownerdom, + $ownername); + unless (&Apache::lonnet::del_dom($reqsnamespace,[$cnum.'_pending'],$dom) eq 'ok') { + $output .= &mt('An error occurred when removing the request for [_1] submitted by [_2] from the pending queue.',$instcode,$ownername.':'.$ownerdom).$linefeed; + } + } + } + } + } + foreach my $key (sort(keys(%newcids))) { + if (ref($newcids{$key}) eq 'ARRAY') { + $output .= "created course from queued request: $key - ".join(', ',@{$newcids{$key}}).$linefeed; + my $newcourse = &LONCAPA::escape($key.':'.$newcids{$key}); + } + } + unless ($context eq 'auto') { + if (keys(%stillpending) > 0) { + $output .= '<form method="post" name="changequeue" action="/adm/createcourse" />'."\n". + '<input type="hidden" name="action" value="'.$env{'form.action'}.'" />'."\n". + '<input type="hidden" name="phase" value="requestchange" />'. + '<p>'.&mt('For the following requests, the requestor could [_1]not[_2] be validated as official course personnel, so the request remains in the pending queue.','<b>','</b>').'<br />'.&mt('Requests may be left in the queue, or you can manually approve or reject them.').'</p>'. + &build_queue_display($dom,'stillpending',\%stillpending). + '<br /><input type="hidden" name="queue" value="pending" />'."\n". + '<input type="submit" name="processqueue" value="'.&mt('Save').'" />'. + '</form>'; + } + } + return $output; +} + +sub postprocess_crsenv { + my ($dom,$cnum,$postprocessenv) = @_; + if (ref($postprocessenv) eq 'HASH') { + my $cid = $dom.'_'.$cnum; + my %settablecrsenv = ( + 'internal.selfenroll_types' => 1, + 'internal.selfenroll_registered' => 1, + 'internal.selfenroll_section' => 1, + 'internal.selfenroll_start_access' => 1, + 'internal.selfenroll_end_access' => 1, + 'internal.selfenroll_limit' => 1, + 'internal.selfenroll_cap' => 1, + 'internal.selfenroll_approval' => 1, + 'internal.selfenroll_notifylist' => 1, + ); + my %needcrsidput = ( + 'internal.selfenroll_types' => 1, + 'internal.selfenroll_start_date' => 1, + 'internal.selfenroll_end_date' => 1, + ); + my (@needupdate,%newcrsenv); + foreach my $key (keys(%{$postprocessenv})) { + if ($settablecrsenv{$key}) { + $newcrsenv{$key} = $postprocessenv->{$key}; + if ($needcrsidput{$key}) { + push(@needupdate,$key); + } + } + if (keys(%newcrsenv)) { + my $putresult = &Apache::lonnet::put('environment',\%newcrsenv,$dom,$cnum); + if ($putresult eq 'ok') { + if (@needupdate) { + my %crsinfo = + &Apache::lonnet::courseiddump($dom,'.',1,'.','.',$cnum,undef,undef,'.'); + if (ref($crsinfo{$cid}) eq 'HASH') { + foreach my $key (@needupdate) { + $crsinfo{$cid}{$key} = $newcrsenv{$key}; + } + my $chome = &Apache::lonnet::homeserver($cnum,$dom); + &Apache::lonnet::courseidput($dom,\%crsinfo,$chome,'notime'); + } + } + } + } + } + } + return; +} + + +sub requestcourses_validation_types { + my @items = ('url','fields','button','markup'); + my %names = &Apache::lonlocal::texthash ( + url => 'Web address of validation server/script', + fields => 'Form fields to send to validator', + button => 'Text for validation button', + markup => 'Validation description (HTML)', + ); + my @fields = ('owner','course','coursetype','description'); + return (\@items,\%names,\@fields); +} + +sub is_active_author { + if ($env{'user.role.au./'.$env{'user.domain'}.'/'} =~ /^(\d*)\.(\d*)$/) { + if ((!$1 || $1 < time) && + (!$2 || $2 > time)) { + return 1; + } + } +} + +sub author_prompt { + my ($is_active_author,$offer_author); + if ($env{'environment.canrequest.author'}) { + unless (&is_active_author()) { + unless (&reqauthor_check() =~ /^approval:\d+$/) { + $offer_author = 1; + } + } + } + return $offer_author; +} + +sub reqauthor_check { + my $queued = $env{'environment.requestauthorqueued'}; + my %reqauthor = &Apache::lonnet::get('requestauthor',['author_status','author'], + $env{'user.domain'},$env{'user.name'}); + my $reqstatus = $reqauthor{'author_status'}; + if (($reqstatus eq '' && $queued ne '') || + ($env{'environment.requestauthorqueued'} !~ /^\Q$reqstatus\E/)) { + if (ref($reqauthor{'author'}) eq 'HASH') { + $queued = $reqstatus.':'.$reqauthor{'author'}{'timestamp'}; + } else { + undef($queued); + } + &Apache::lonnet::appenv({'environment.requestauthorqueued' => $queued}); + } + return $queued; +} + +sub process_reqauthor { + my ($dispositionref,$updateref) = @_; + if (&is_active_author()) { + return '<span class="LC_warning">'. + &mt('An Authoring Space has already been assigned to you.').'<br />'. + &mt('Please select the Author role from your [_1]roles page[_2].','<a href="/adm/roles">', + '</a>').'</span>'; + } + unless ($env{'environment.canrequest.author'}) { + return '<span class="LC_warning">'. + &mt('You do not currently have rights to request an Authoring Space.').'<br />'. + &mt('Please contact the [_1]helpdesk[_2] for assistance.','<a href="/adm/helpdesk">', + '</a>').'</span>'; + } + my $queued = &reqauthor_check(); + if ($queued =~ /^approval:(\d+)$/) { + my $timestamp = $1; + return '<span class="LC_info">'. + &mt('A request for Authoring Space submitted on [_1] is awaiting approval', + &Apache::lonlocal::locallocaltime($timestamp)). + '</span>'; + } elsif ($queued =~ /^approved:(\d+)$/) { + my $timestamp = $1; + my %roleshash = &Apache::lonnet::get_my_roles($env{'user.name'},$env{'user.domain'},'userroles', + ['active'],['au'],[$env{'user.domain'}]); + if (keys(%roleshash) > 0) { + return '<span class="LC_info">'. + &mt('A request for Authoring Space submitted on [_1] has been approved.', + &Apache::lonlocal::locallocaltime($timestamp)). + '</span>'; + } + } + my ($output,@inststatuses,%domconfig); + %domconfig = &Apache::lonnet::get_dom('configuration',['requestauthor'], + $env{'user.domain'}); + my $val = &get_processtype('requestauthor',$env{'user.name'},$env{'user.domain'}, + $env{'user.adv'},$env{'user.domain'},undef, + \@inststatuses,\%domconfig); + my $now = time; + if ($val eq 'automatic') { + my $start = $now-1; + if (&Apache::lonnet::assignrole($env{'user.domain'},$env{'user.name'},'/'.$env{'user.domain'}.'/', + 'au',undef,$start,undef,undef,'requestauthor') eq 'ok') { + $output = '<span class="LC_info">'. + &mt('Access to Authoring Space has been activated').'</span><br />'; + &Apache::lonroles::update_session_roles(); + &Apache::lonnet::appenv({'user.update.time' => $now}); + if (ref($updateref)) { + $$updateref = $now; + } + if (ref($dispositionref)) { + $$dispositionref = 'created'; + } + } else { + $output = '<span class="LC_info">'. + &mt('An error occurred while activating your access to Authoring Space'); + } + } elsif ($val eq 'approval') { + my $domconfiguser = &Apache::lonnet::get_domainconfiguser($env{'user.domain'}); + if (&Apache::lonnet::put('requestauthorqueue',{ $env{'user.name'}.'_'.$val => $now }, + $env{'user.domain'},$domconfiguser) eq 'ok') { + my %userrequest = ( + author => { + timestamp => $now, + status => $val, + }, + author_status => $val, + ); + my $req_notifylist; + if (ref($domconfig{'requestauthor'}) eq 'HASH') { + if (ref($domconfig{'requestauthor'}{'notify'}) eq 'HASH') { + my $req_notifylist = $domconfig{'requestauthor'}{'notify'}{'approval'}; + if ($req_notifylist) { + my $fullname = &Apache::loncommon::plainname($env{'user.name'}, + $env{'user.domain'}); + my $sender = $env{'user.name'}.':'.$env{'user.domain'}; + my $domdesc = &Apache::lonnet::domain($env{'user.domain'},'description'); + &send_selfserve_notification($req_notifylist, + "$fullname ($env{'user.name'}:$env{'user.domain'})", + undef,$domdesc,$now,'authorreq',$sender); + } + } + } + my $userresult = + &Apache::lonnet::put('requestauthor',\%userrequest,$env{'user.domain'},$env{'user.name'}); + $output = '<span class="LC_info">'. + &mt('Your request for Authoring Space has been submitted for approval.'). + '</span>'; + &Apache::lonnet::appenv({'environment.requestauthorqueued' => $val.':'.$now}); + } else { + $output = '<span class="LC_info">'. + &mt('An error occurred saving your request for Authoring Space.'). + '</span>'; + } + } + return $output; +} + 1;