--- loncom/publisher/loncfile.pm 2024/05/13 13:55:50 1.128 +++ loncom/publisher/loncfile.pm 2024/08/24 22:09:30 1.129.2.1 @@ -9,7 +9,7 @@ # and displays a page showing the results of the action. # # -# $Id: loncfile.pm,v 1.128 2024/05/13 13:55:50 raeburn Exp $ +# $Id: loncfile.pm,v 1.129.2.1 2024/08/24 22:09:30 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -822,42 +822,19 @@ sub Decompress1 { sub Archive1 { my ($request,$fn) = @_; my @posstypes = qw(problem library sty sequence page task rights meta xml html xhtml htm xhtm css js tex txt gif jpg jpeg png svg other); - my (%location_of,%default,$compstyle); - foreach my $program ('tar','gzip','bzip2','xz','zip') { - foreach my $dir ('/bin/','/usr/bin/','/usr/local/bin/','/sbin/', - '/usr/sbin/') { - if (-x $dir.$program) { - $location_of{$program} = $dir.$program; - last; - } - } - } - my (%defaults,$cancompress,$canarchive); - if (exists($location_of{'tar'})) { - $default{'tar'} = ' checked="checked"'; - $canarchive = 1; - $compstyle = 'block'; - } elsif (exists($location_of{'zip'})) { - $default{'zip'} = ' checked="checked"'; - $canarchive = 1; - $compstyle = 'none'; - } - foreach my $compress ('gzip','bzip2','xz') { - if (exists($location_of{$compress})) { - $default{$compress} = ' checked="checked"'; - $cancompress = 1; - last; - } - } + my (%location_of,%defaults); + my ($compstyle,$canarchive,$cancompress,$numformat,$numcompress,$defext) = + &archive_tools(\%location_of,\%defaults); if (!$canarchive) { $request->print('

'. - &mt('This LON-CAPA instance does not seem to have either tar or zip installed.').'

'. + &mt('This LON-CAPA instance does not seem to have either tar or zip installed.').'

'."\n". ''. &mt('At least one of the two is needed in order to be able to create an archive file for: [_1].', - &display($fn)). + &display($fn))."\n". ''); } elsif (-e $fn) { - $request->print(&Apache::lonhtmlcommon::start_pick_box(). + $request->print(''."\n". + &Apache::lonhtmlcommon::start_pick_box(). &Apache::lonhtmlcommon::row_title(&mt('Directory')). &display($fn). &Apache::lonhtmlcommon::row_closure(). @@ -909,7 +886,7 @@ sub Archive1 { if (exists($location_of{$possfmt})) { $request->print(''. '   '); } } @@ -921,7 +898,8 @@ sub Archive1 { foreach my $compress ('gzip','bzip2','xz') { if (exists($location_of{$compress})) { $request->print('  '); + $defaults{$compress}.' onclick="setArchiveExt(this.form);" />'. + $compress.'  '); } } } else { @@ -929,9 +907,16 @@ sub Archive1 { &mt('This LON-CAPA instance does not seem to have gzip, bzip2 or xz installed.'). '
'.&mt('No compression will be used.').''); } - $request->print(''. + $request->print(''."\n". + ''."\n". &Apache::lonhtmlcommon::row_closure(1). - &Apache::lonhtmlcommon::end_pick_box() + &Apache::lonhtmlcommon::end_pick_box().'
'."\n" ); &CloseForm1($request, $fn); } else { @@ -941,6 +926,110 @@ sub Archive1 { .'

' ); } + return; +} + +sub archive_tools { + my ($location_of,$defaults) = @_; + my ($compstyle,$canarchive,$cancompress,$numformat,$numcompress,$defext); + ($numformat,$numcompress) = (0,0); + if ((ref($location_of) eq 'HASH') && (ref($defaults) eq 'HASH')) { + foreach my $program ('tar','gzip','bzip2','xz','zip') { + foreach my $dir ('/bin/','/usr/bin/','/usr/local/bin/','/sbin/', + '/usr/sbin/') { + if (-x $dir.$program) { + $location_of->{$program} = $dir.$program; + last; + } + } + } + foreach my $format ('tar','zip') { + if (exists($location_of->{$format})) { + unless ($canarchive) { + $defext = $format; + $defaults->{$format} = ' checked="checked"'; + if ($format eq 'tar') { + $compstyle = 'block'; + } else { + $compstyle = 'none'; + } + } + $canarchive = 1; + $numformat ++; + } + } + foreach my $compress ('gzip','bzip2','xz') { + if (exists($location_of->{$compress})) { + $numcompress ++; + unless ($cancompress) { + if ($defext eq 'tar') { + if ($compress eq 'gzip') { + $defext .= '.gz'; + } elsif ($compress eq 'bzip2') { + $defext .= '.bz2'; + } else { + $defext .= ".$compress"; + } + } + $defaults->{$compress} = ' checked="checked"'; + $cancompress = 1; + } + } + } + } + if (wantarray) { + return ($compstyle,$canarchive,$cancompress,$numformat,$numcompress,$defext); + } else { + return $defext; + } +} + +sub archive_in_progress { + my ($earlyout,$idnum); + if ($env{'cgi.author.archive'} =~ /^(\d+)_\d+_\d+$/) { + my $timestamp = $1; + $idnum = $env{'cgi.author.archive'}; + if (exists($env{'cgi.'.$idnum.'.archive'})) { + my $hashref = &Apache::lonnet::thaw_unescape($env{'cgi.'.$idnum.'.archive'}); + my $lonprtdir = $Apache::lonnet::perlvar{'lonPrtDir'}; + if (-e $lonprtdir.'/'.$env{'user.name'}.'_'.$env{'user.domain'}.'_archive_'.$idnum.'.txt') { + $earlyout = $timestamp; + } elsif (ref($hashref) eq 'HASH') { + my $suffix = $hashref->{'extension'}; + if (-e $lonprtdir.'/'.$env{'user.name'}.'_'.$env{'user.domain'}.'_archive_'.$idnum.$suffix) { + $earlyout = $timestamp; + } + } + unless ($earlyout) { + &Apache::lonnet::delenv('cgi.'.$idnum.'.archive'); + &Apache::lonnet::delenv('cgi.author.archive'); + } + } else { + &Apache::lonnet::delenv('cgi.author.archive'); + } + } + return ($earlyout,$idnum); +} + +sub cancel_archive_form { + my ($r,$title,$fname,$earlyout,$idnum) = @_; + $r->print('

'.$title.'

'."\n". + '
'."\n". + ''."\n". + ''."\n". + '

'.&mt('Each author may only have one archive request in process at a time.')."\n".'

'."\n". + '

'.&mt('Remove existing archive request?').' '."\n". + ''. + (' 'x2)."\n". + '

'."\n". + '
'); } =pod @@ -1118,10 +1207,10 @@ sub phaseone { '

'); return; } - $r->print(''. - ''. - ''. - ''); + $r->print(''."\n". + ''."\n". + ''."\n". + ''."\n"); if ($env{'form.action'} eq 'newfile' || $env{'form.action'} eq 'newhtmlfile' || @@ -1437,10 +1526,22 @@ sub decompress2 { } sub Archive2 { - my ($r,$name,$udom,$fn,$identifier) = @_; + my ($r,$uname,$udom,$fn,$identifier) = @_; my %options = ( dir => $fn, + uname => $uname, + udom => $udom, ); + if ($env{'form.adload'}) { + $options{'adload'} = 1; + if ($env{'form.archivefname'} ne '') { + $env{'form.archivefname'} =~ s{\.+}{.}g; + $options{'fname'} = $env{'form.archivefname'}; + } + if ($env{'form.archiveext'} ne '') { + $options{'extension'} = $env{'form.archiveext'}; + } + } my @filetypes = qw(problem library sty sequence page task rights meta xml html xhtml htm xhtm css js tex txt gif jpg jpeg png svg other); my (@include,%oktypes); map { $oktypes{$_} = 1; } @filetypes; @@ -1473,10 +1574,40 @@ sub Archive2 { } my $key = 'cgi.'.$identifier.'.archive'; my $storestring = &Apache::lonnet::freeze_escape(\%options); - &Apache::lonnet::appenv({$key => $storestring}); + &Apache::lonnet::appenv({$key => $storestring, + 'cgi.author.archive' => $identifier}); return 1; } +sub Archive3 { + my ($hashref) = @_; + if (ref($hashref) eq 'HASH') { + if (($hashref->{'uname'} eq $env{'user.name'}) && + ($hashref->{'udom'} eq $env{'user.domain'}) && + ($env{'environment.canarchive'}) && + ($env{'form.delarchive'})) { + my $filesdest = $Apache::lonnet::perlvar{'lonPrtDir'}.'/'.$env{'user.name'}.'_'.$env{'user.domain'}.'_archive_'.$env{'form.delarchive'}; + if (-e $filesdest) { + my $size = (stat($filesdest))[7]; + if (unlink($filesdest)) { + my ($identifier,$suffix) = split(/\./,$env{'form.delarchive'},2); + if (($identifier) && (exists($env{'cgi.'.$identifier.'.archive'}))) { + my $delres = &Apache::lonnet::delenv('cgi.'.$identifier.'.archive'); + if (($delres eq 'ok') && + (exists($env{'cgi.author.archive'})) && + ($env{'cgi.author.archive'} eq $identifier)) { + &Apache::lonnet::authorarchivelog($hashref,$size,$filesdest,'delete'); + &Apache::lonnet::delenv('cgi.author.archive'); + } + } + return 1; + } + } + } + } + return 0; +} + =pod =item phasetwo($r, $fn, $uname, $udom,$identifier) @@ -1624,14 +1755,26 @@ sub handler { # # Determine the root filename # This could come in as "filename", which actually is a URL, or -# as "qualifiedfilename", which is indeed a real filename in filesystem +# as "qualifiedfilename", which is indeed a real filename in filesystem, +# or in value of decompress form element, or need to be extracted +# from %env from hashref retrieved for cgi..archive key, where id +# is a unique cgi_id created when an Author creates an archive of +# Authoring Space for download. # - my $fn; + my ($fn,$archiveref); if ($env{'form.filename'}) { &Debug($r, "test: $env{'form.filename'}"); $fn=&unescape($env{'form.filename'}); $fn=&URLToPath($fn); + } elsif ($env{'form.delarchive'}) { + my ($delarchive,$suffix) = split(/\./,$env{'form.delarchive'}); + if (($delarchive) && (exists($env{'cgi.'.$delarchive.'.archive'}))) { + $archiveref = &Apache::lonnet::thaw_unescape($env{'cgi.'.$delarchive.'.archive'}); + if (ref($archiveref) eq 'HASH') { + $fn = $archiveref->{'dir'}; + } + } } elsif($ENV{'QUERY_STRING'} && $env{'form.phase'} ne 'two') { #Just hijack the script only the first time around to inject the #correct information for further processing @@ -1666,12 +1809,19 @@ sub handler { $r->filename); return HTTP_NOT_ACCEPTABLE; } - + if (($env{'form.delarchive'}) && + ($env{'environment.canarchive'})) { + &Apache::loncommon::content_type($r,'text/plain'); + $r->send_http_header; + $r->print(&Archive3($archiveref)); + return OK; + } &Apache::loncommon::content_type($r,'text/html'); $r->send_http_header; - my ($js,$identifier); +# Declarations for items used for directory archive requests + my ($js,$identifier,$defext,$archive_earlyout,$archive_idnum); my $args = {}; if (($env{'form.action'} eq 'newdir') && ($env{'form.phase'} eq 'two') && @@ -1691,15 +1841,60 @@ function writeDone() { ENDJS $args->{'add_entries'} = { onload => "writeDone()" }; } elsif (($env{'form.action'} eq 'archive') && - ($env{'environment.authorarchive'})) { - if ($env{'form.phase'} eq 'two') { - $identifier = &Apache::loncommon::get_cgi_id(); - $args->{'redirect'} = [0,"/cgi-bin/archive.pl?$identifier"]; - } else { - my $check_uncheck_js = &Apache::loncommon::check_uncheck_jscript(); + ($env{'environment.canarchive'})) { +# Check if author already has an archive request in process + ($archive_earlyout,$archive_idnum) = &archive_in_progress(); +# Check if archive request was in process which author wishes to terminate + if ($env{'form.remove_archive_request'}) { + if ($env{'form.remove_archive_request'} eq $archive_idnum) { + if (exists($env{'cgi.'.$archive_idnum.'.archive'})) { + my $archiveref = &Apache::lonnet::thaw_unescape($env{'cgi.'.$archive_idnum.'.archive'}); + if (ref($archiveref) eq 'HASH') { + $env{'form.delarchive'} = $archive_idnum.$archiveref->{'extension'}; + if (&Archive3($archiveref)) { + ($archive_earlyout,$archive_idnum) = &archive_in_progress(); + } + delete($env{'form.delarchive'}); + } + } + } + } + if ($archive_earlyout) { + my $conftext = + &mt('Removing an existing request will terminate an active download of the archive file.'); + &js_escape(\$conftext); $js = <<"ENDJS"; + +ENDJS + } else { + if ($env{'form.phase'} eq 'two') { + $identifier = &Apache::loncommon::get_cgi_id(); + $args->{'redirect'} = [0.1,"/cgi-bin/archive.pl?$identifier"]; + } else { + my (%location_of,%defaults); + $defext = &archive_tools(\%location_of,\%defaults); + my $check_uncheck_js = &Apache::loncommon::check_uncheck_jscript(); + $js = <<"ENDJS"; + ENDJS - $args->{'add_entries'} = { onload => "resetForm()" }; + $args->{'add_entries'} = { onload => "resetForm()" }; + } } } my $londocroot = $r->dir_config('lonDocRoot'); @@ -1762,19 +2027,9 @@ ENDJS $trailfile =~ s{^/(priv/)}{$londocroot/$1}; # Breadcrumbs - my $crsauthor; my $text = 'Authoring Space'; my $title = 'Authoring Space File Operation', my $href = &Apache::loncommon::authorspace(&url($fn)); - if ($env{'request.course.id'}) { - my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'}; - my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'}; - if ($href eq "/priv/$cdom/$cnum/") { - $text = 'Course Authoring Space'; - $title = 'Course Authoring Space File Operation', - $crsauthor = 1; - } - } &Apache::lonhtmlcommon::clear_breadcrumbs(); &Apache::lonhtmlcommon::add_breadcrumb({ 'text' => $text, @@ -1797,12 +2052,10 @@ ENDJS } if (($uname ne $env{'user.name'}) || ($udom ne $env{'user.domain'})) { - unless ($crsauthor) { - $r->print('

' - .&mt('Co-Author [_1]',$uname.':'.$udom) - .'

' - ); - } + $r->print('

' + .&mt('Co-Author [_1]',$uname.':'.$udom) + .'

' + ); } @@ -1827,34 +2080,17 @@ ENDJS 'Select Action' => 'New Resource', ); if ($action{$env{'form.action'}}) { - if ($crsauthor) { - my @disallowed = qw(page sequence rights library); - my $newtype; - if ($env{'form.action'} =~ /^new(\w+)file$/) { - $newtype = $1; - } elsif ($env{'form.action'} eq 'newfile') { - ($newtype) = ($env{'form.newfilename'} =~ m{\.([^/.]+)$}); - $newtype = lc($newtype); - } - if (($newtype ne '') && - (grep(/^\Q$newtype\E$/,@disallowed))) { - $r->print('

' - .&mt('Creation of a new file of type: [_1] is not permitted in Course Authoring Space',$newtype) - .'

' - .&Apache::loncommon::end_page() - ); - return OK; - } - if ($env{'form.action'} eq 'archive') { - $r->print('

'.&mt('Location').': '.&display($fn).'

'."\n". - '

'. - &mt('Export to an archive file is not permitted in Course Authoring Space'). - '

'."\n". - &Apache::loncommon::end_page()); - return OK; - } - } elsif ($env{'form.action'} eq 'archive') { - unless ($env{'environment.authorarchive'}) { + if ($env{'form.action'} eq 'archive') { + if ($env{'environment.canarchive'}) { + if ($archive_earlyout) { + my $fname = &url($fn); + my $title = $action{$env{'form.action'}}; + &cancel_archive_form($r,$title,$fname,$archive_earlyout,$archive_idnum); + &CloseForm1($r,$fn); + $r->print(&Apache::loncommon::end_page()); + return OK; + } + } else { $r->print('

'.&mt('Location').': '.&display($fn).'

'."\n". '

'. &mt('You do not have permission to export to an archive file in this Authoring Space'). @@ -1863,7 +2099,7 @@ ENDJS return OK; } } - $r->print('

'.$action{$env{'form.action'}}.'

'); + $r->print('

'.$action{$env{'form.action'}}.'

'."\n"); } else { $r->print('

' .&mt('Unknown Action: [_1]',$env{'form.action'})