--- loncom/interface/londocs.pm 2024/12/22 03:12:53 1.713
+++ loncom/interface/londocs.pm 2025/01/07 21:01:37 1.722
@@ -1,7 +1,7 @@
# The LearningOnline Network
# Documents
#
-# $Id: londocs.pm,v 1.713 2024/12/22 03:12:53 raeburn Exp $
+# $Id: londocs.pm,v 1.722 2025/01/07 21:01:37 raeburn Exp $
#
# Copyright Michigan State University Board of Trustees
#
@@ -579,7 +579,8 @@ $contents{webreferences}.'
$title=~s/[\/\s]+/\_/gs;
$title=&clean($title);
my $formname = 'dumpdoc';
- my $preamble = &authorspace_selector($r,$formname,$home,$title,%outhash);
+ my $preamble = &authorspace_selector($r,$formname,$home,$title,%outhash).
+ '
'."\n";
my %uploadedfiles;
&tiehash();
foreach my $file (&Apache::lonclonecourse::crsdirlist($origcrsid,'userfiles')) {
@@ -643,8 +644,8 @@ sub authorspace_selector {
}
$preamble .= ''.
'
'."\n";
+ ''."\n".
+ ''."\n";
return $preamble;
}
@@ -694,9 +695,70 @@ sub recurse_html {
sub copycrsauthored {
my ($r,$coursenum,$coursedom,$coursehome,$readonly) = @_;
- my ($starthash,$js);
+ my ($starthash,$js,$title,$formname);
+ my %origcrsdata=&Apache::lonnet::coursedescription($env{'request.course.id'});
+ $title=$origcrsdata{'description'};
+ $title=~s/[\/\s]+/\_/gs;
+ $title=&clean($title);
+ my ($home,$other,%outhash)=&authorhosts();
unless (($env{'form.authorspace'}) && ($env{'form.authorfolder'}=~/\w/)) {
- $js = <<"ENDJS";
+ my %js_lt;
+ $formname = 'copycrsauthored';
+ if ($home) {
+ %js_lt =
+ &Apache::lonlocal::texthash(
+ yomu => 'You must select an Authoring Space',
+ whco => 'When Copyright set to "custom", URL of a published rights file is needed.',
+ );
+ &js_escape(\%js_lt);
+ }
+ if ($home > 1) {
+ $js = <<"ENDJS";
+
+
+ENDJS
+ } elsif ($home) {
+ $js = <<"ENDJS";
+
+
+ENDJS
+ }
+ $js .= <<"ENDJS";
ENDJS
+
+ $js .= "\n".&Apache::lonhtmlcommon::scripttag(&Apache::loncommon::browser_and_searcher_javascript())."\n";
$starthash = {
- add_entries => {'onload' => "hide_searching();"},
+ add_entries => {'onload' => "hide_searching(); init_copycrs_form();"},
};
}
$r->print(&Apache::loncommon::start_page('Copy from Course Authoring to User Authoring',$js,$starthash)."\n".
&Apache::lonhtmlcommon::breadcrumbs('Copy from Course Authoring Space')."\n");
$r->print(&startContentScreen('tools'));
- my ($home,$other,%outhash)=&authorhosts();
unless ($home) {
$r->print(''.&mt('No author or co-author roles on this server.').'
');
$r->print(&endContentScreen());
return '';
}
- my %origcrsdata=&Apache::lonnet::coursedescription($env{'request.course.id'});
+ my $docroot = $r->dir_config('lonDocRoot');
+ my $is_course_home;
+ my @ids=&Apache::lonnet::current_machine_ids();
+ if (($coursehome ne '') && (grep(/^\Q$coursehome\E$/,@ids))) {
+ $is_course_home = 1;
+ }
my $exclude = &Apache::lonnet::priv_exclude();
my $srcurl = "/priv/$coursedom/$coursenum";
- my $srctop = $r->dir_config('lonDocRoot').$srcurl;
+ my $srctop = $docroot.$srcurl;
+ my $resurl = "/res/$coursedom/$coursenum";
+ my $res_exclude = &Apache::lonnet::res_exclude();
if (($env{'form.authorspace'}) && ($env{'form.authorfolder'}=~/\w/)) {
$r->print(''.&mt('Copying Files and/or Sub-directories').'
');
if ($readonly) {
@@ -743,7 +824,8 @@ ENDJS
}
my ($ca,$cd)=split(/\:/,$env{'form.authorspace'});
my $desturl = "/priv/$cd/$ca";
- my $desttop = $r->dir_config('lonDocRoot').$desturl;
+ my $destresurl = "/res/$cd/$ca";
+ my $desttop = $docroot.$desturl;
my $subdir = &clean($env{'form.authorfolder'});
$subdir = &cleandir($subdir);
if ($subdir eq '') {
@@ -755,16 +837,11 @@ ENDJS
&endContentScreen());
return '';
}
- my $is_course_home;
- my @ids=&Apache::lonnet::current_machine_ids();
- if (($coursehome ne '') && (grep(/^\Q$coursehome\E$/,@ids))) {
- $is_course_home = 1;
- }
my (%tocopy,%dirs_to_make,%files_to_copy);
map { $tocopy{&unescape($_)} = 1; } &Apache::loncommon::get_env_multiple('form.copytouser');
if (keys(%tocopy)) {
my (%subdirs,%files);
- &Apache::lonnet::recursedirs($home,1,undef,$exclude,0,0,$srcurl,'',\%subdirs,\%files);
+ &Apache::lonnet::recursedirs($is_course_home,1,undef,$exclude,0,0,$srcurl,'',\%subdirs,\%files);
foreach my $possible (sort(keys(%tocopy))) {
if ($possible =~ m{/$}) {
my $possdir = $possible;
@@ -805,289 +882,284 @@ ENDJS
return '';
}
if (keys(%tocopy)) {
- my ($notopdir,%newdir,%newfile,%checkdeps);
+ my (%resdirs,%resfiles);
+ &Apache::lonnet::recursedirs($is_course_home,1,undef,$res_exclude,0,0,$resurl,'',\%resdirs,\%resfiles);
+ my ($notopdir,%newdir,%newfile,%checkdeps,%newresfile);
$r->print(''.&mt('Copy to: [_1]',
''.$desturl.'/'.$subdir.'').
'
'."\n");
- unless ($is_course_home) {
- $r->print(''.
- &endContentScreen());
- return '';
- }
if (keys(%dirs_to_make)) {
- if ($is_course_home) {
- unless (-e $desttop.'/'.$subdir) {
- mkdir($desttop.'/'.$subdir,0755);
- }
- if (-e $desttop.'/'.$subdir) {
- foreach my $dir (sort(keys(%dirs_to_make))) {
- my @dirs=split(/\//,$dir);
- my $path="$desttop/$subdir";
- my $makepath=$path;
- my $fail;
- for (my $i=0;$i<@dirs;$i++) {
- $makepath.='/'.$dirs[$i];
- unless (-e $makepath) {
- unless (mkdir($makepath,0755)) {
- $fail = 1;
- last;
- }
- if (($i == scalar(@dirs)-1) && (!$fail)) {
- $newdir{$dir} = 1;
- }
+ unless (-e $desttop.'/'.$subdir) {
+ mkdir($desttop.'/'.$subdir,0755);
+ }
+ if (-e $desttop.'/'.$subdir) {
+ foreach my $dir (sort(keys(%dirs_to_make))) {
+ my @dirs=split(/\//,$dir);
+ my $path="$desttop/$subdir";
+ my $makepath=$path;
+ my $fail;
+ for (my $i=0;$i<@dirs;$i++) {
+ $makepath.='/'.$dirs[$i];
+ unless (-e $makepath) {
+ unless (mkdir($makepath,0755)) {
+ $fail = 1;
+ last;
+ }
+ if (($i == scalar(@dirs)-1) && (!$fail)) {
+ $newdir{$dir} = 1;
}
- }
- if ($fail) {
- $r->print('
'.&mt('Target directory: [_1] does not exist, and could not be created.',
- ''.$desturl.'/'.$subdir.'/'.$dir.'').
- '
'."\n");
}
}
- } else {
- $notopdir = 1;
+ if ($fail) {
+ $r->print(''.&mt('Target directory: [_1] does not exist, and could not be created.',
+ ''.$desturl.'/'.$subdir.'/'.$dir.'').
+ '
'."\n");
+ }
}
+ } else {
+ $notopdir = 1;
}
}
if (keys(%files_to_copy)) {
- if ($is_course_home) {
- unless (-e $desttop.'/'.$subdir) {
- mkdir($desttop.'/'.$subdir,0755);
- }
- if (-e $desttop.'/'.$subdir) {
- my $num = 0;
- foreach my $file (keys(%files_to_copy)) {
- my ($fail,$dup,$dir_is_file,$src,$dest,$path,$fname);
- if ($file =~ m{/}) {
- ($path,$fname) = ($file =~ m{^(.+)/([^/]+)$});
- if (-d "$desttop/$subdir/$path") {
- if (-e "$desttop/$subdir/$path/$fname") {
- $dup = 1;
+ unless (-e $desttop.'/'.$subdir) {
+ mkdir($desttop.'/'.$subdir,0755);
+ }
+ if (-e $desttop.'/'.$subdir) {
+ my $num = 0;
+ my ($copyright,$customdistfile);
+ if ($env{'form.copyright'} eq 'default' || $env{'form.copyright'} eq 'domain' || $env{'form.copyright'} eq 'public') {
+ $copyright = $env{'form.copyright'};
+ } elsif ($env{'form.copyright'} eq 'custom') {
+ if ($env{'form.customrights'} =~ m{^/res/$match_domain/$match_username/.+\.rights$}) {
+ my ($rightsdom,$rightsuname) = ($1,$2);
+ my $rightshome = &Apache::lonnet::homeserver($rightsdom,$rightsuname);
+ if (($rightshome eq 'no_host') || ($rightshome eq '')) {
+ $copyright = 'default';
+ } elsif (grep(/^\Q$rightshome\E$/,@ids)) {
+ if (-e $docroot.$env{'form.customrights'}) {
+ $copyright = 'custom';
+ $customdistfile = $env{'form.customrights'};
+ } else {
+ $copyright = 'default';
+ }
+ } else {
+ my $rightsfile = &Apache::lonnet::filelocation('',$env{'form.customrights'});
+ unless (&Apache::lonnet::getfile($rightsfile) eq '-1') {
+ $customdistfile = $env{'form.customrights'};
+ }
+ }
+ }
+ }
+ my $sourceavail;
+ if ($env{'form.sourceavail'} =~ /^(open|closed)$/) {
+ $sourceavail = $env{'form.sourceavail'};
+ }
+ my $respublish;
+ if ($env{'form.respublish'}) {
+ $respublish = 1;
+ }
+ my $nokeyref = &Apache::lonpublisher::getnokey($r->dir_config('lonIncludes'));
+ foreach my $file (keys(%files_to_copy)) {
+ my ($fail,$dup,$dir_is_file,$src,$dest,$path,$fname);
+ if ($file =~ m{/}) {
+ ($path,$fname) = ($file =~ m{^(.+)/([^/]+)$});
+ if (-d "$desttop/$subdir/$path") {
+ if (-e "$desttop/$subdir/$path/$fname") {
+ $dup = 1;
+ } else {
+ $src = "$srctop/$path/$fname";
+ $dest = "$desttop/$subdir/$path/$fname";
+ }
+ } elsif (-f "$desttop/$subdir/$path") {
+ $dir_is_file = 1;
+ } else {
+ $fail = 1;
+ }
+ } elsif (-e "$desttop/$subdir/$file") {
+ $dup = 1;
+ } else {
+ $src = "$srctop/$file";
+ $dest = "$desttop/$subdir/$file";
+ $fname = $file;
+ }
+ if ($fail) {
+ $r->print(''.&mt('Target directory: [_1] does not exist, and could not be created.',
+ ''.$desturl.'/'.$subdir.'/'.$path.'').
+ '
'."\n");
+ } elsif ($dup) {
+ $r->print(''.&mt('Target file: [_1] already exists -- not overwriting.',
+ ''.$desturl.'/'.$subdir.'/'.$file.'').
+ '
'."\n");
+ } elsif ($dir_is_file) {
+ $r->print(''.&mt('Target directory: [_1] name is already in a use for a file -- not overwriting.',
+ ''.$desturl.'/'.$subdir.'/'.$file.'').
+ '
'."\n");
+ } elsif (($src ne '') && ($dest ne '')) {
+ my $ressrc = $docroot.$resurl.'/'.$file;
+ my $ressrcmeta = $ressrc.'.meta';
+ my ($ext) = ($file =~ /\.(\w+)$/);
+ my $embstyle=&Apache::loncommon::fileembstyle($ext);
+ my ($getres,$getresmeta);
+ if ($respublish) {
+ if ($path eq '') {
+ if ((ref($resfiles{'/'}) eq 'HASH') &&
+ (exists($resfiles{'/'}{$fname}))) {
+ $getres = 1;
+ $getresmeta = 1;
+ }
+ } elsif ((ref($resfiles{$path}) eq 'HASH') &&
+ (exists($resfiles{$path}{$fname}))) {
+ $getres = 1;
+ $getresmeta = 1;
+ }
+ }
+ if ($is_course_home) {
+ my ($needpriv,$needprivmeta);
+ if ($respublish) {
+ if ($getres) {
+ if (&Apache::londiff::are_different_files($src,$ressrc)) {
+ $needpriv = 1;
+ if (&File::Copy::copy($ressrc,$dest)) {
+ if ($embstyle eq 'ssi') {
+ &crsres_fixup($dest,$coursenum,$coursedom,$ca,$cd);
+ }
+ }
+ } else {
+ if (&File::Copy::copy($src,$dest)) {
+ $newfile{$file} = $desturl.'/'.$subdir.'/'.$file;
+ if ($embstyle eq 'ssi') {
+ &crsres_fixup($dest,$coursenum,$coursedom,$ca,$cd,$subdir);
+ }
+ }
+ }
} else {
- $src = "$srctop/$path/$fname";
- $dest = "$desttop/$subdir/$path/$fname";
+ $needpriv = 1;
+ }
+ if ($getresmeta) {
+ if ((-e $src.'.meta') && (!-e $dest.'.meta')) {
+ if (&Apache::londiff::are_different_files($src.'.meta',$ressrc.'.meta')) {
+ if (&File::Copy::copy($ressrc.'.meta',$dest.'.meta')) {
+ &crsres_fixup_meta($dest,$coursenum,$coursedom,$ca,$cd,$copyright,
+ $customdistfile,$sourceavail,\%checkdeps);
+ }
+ $needprivmeta = 1;
+ } else {
+ if (&File::Copy::copy($src.'.meta',$dest.'.meta')) {
+ &crsres_fixup_meta($dest,$coursenum,$coursedom,$ca,$cd,$copyright,
+ $customdistfile,$sourceavail,\%checkdeps);
+ }
+ }
+ }
+ }
+ if ($getres) {
+ my $destresfile = $docroot.$destresurl.'/'.$subdir.'/'.$file;
+ if (-e $dest) {
+ my $output = &Apache::lonpublisher::batchpublish($r,$dest,$destresfile,$nokeyref,1);
+ if (-e $destresfile) {
+ $newresfile{$file} = $destresurl.'/'.$subdir.'/'.$file;
+ }
+ }
}
- } elsif (-f "$desttop/$subdir/$path") {
- $dir_is_file = 1;
} else {
- $fail = 1;
+ $needpriv = 1;
+ if ((-e $src.'.meta') && (!-e $dest.'.meta')) {
+ $needprivmeta = 1;
+ }
+ }
+ if ($needpriv) {
+ if (&File::Copy::copy($src,$dest)) {
+ $newfile{$file} = $desturl.'/'.$subdir.'/'.$file;
+ if ($embstyle eq 'ssi') {
+ &crsres_fixup($dest,$coursenum,$coursedom,$ca,$cd,$subdir);
+ }
+ }
+ }
+ if ($needprivmeta) {
+ if (&File::Copy::copy($src.'.meta',$dest.'.meta')) {
+ &crsres_fixup_meta($dest,$coursenum,$coursedom,$ca,$cd,$copyright,
+ $customdistfile,$sourceavail,\%checkdeps);
+ }
}
- } elsif (-e "$desttop/$subdir/$file") {
- $dup = 1;
} else {
- $src = "$srctop/$file";
- $dest = "$desttop/$subdir/$file";
- $fname = $file;
- }
- if ($fail) {
- $r->print(''.&mt('Target directory: [_1] does not exist, and could not be created.',
- ''.$desturl.'/'.$subdir.'/'.$path.'').
- '
'."\n");
- } elsif ($dup) {
- $r->print(''.&mt('Target file: [_1] already exists -- not overwriting.',
- ''.$desturl.'/'.$subdir.'/'.$file.'').
- '
'."\n");
- } elsif ($dir_is_file) {
- $r->print(''.&mt('Target directory: [_1] name is already in a use for a file -- not overwriting.',
- ''.$desturl.'/'.$subdir.'/'.$file.'').
- '
'."\n");
- } elsif (($src ne '') && ($dest ne '')) {
- if (&File::Copy::copy($src,$dest)) {
- $newfile{$file} = 1;
- if ((-e $src.'.meta') && (!-e $dest.'.meta')) {
- if (&File::Copy::copy($src.'.meta',$dest.'.meta')) {
- if (open(my $fh,'<',$dest.'.meta')) {
- my ($output,$now);
- $now = time;
- while (my $line=<$fh>) {
- chomp($line);
- if ($line eq "$coursenum:$coursedom") {
- $output .= "$ca:$cd\n";
- } elsif ($line eq 'custom') {
- $output .= "default\n";
- } elsif ($line =~ m{^\d+$}) {
- $output .= "$now\n";
- } elsif ($line eq "/res/$coursedom/$coursenum/default.rights") {
- $output .= "\n";
- } elsif ($line eq "$coursedom") {
- $output .= "$cd\n";
- } elsif ($line =~ m{^\d+$}) {
- $output .= "$now\n";
- } elsif ($line =~ m{^$match_username:$match_domain$}) {
- $output .= "$env{'user.name'}:$env{'user.domain'}\n";
- } elsif ($line eq "$coursenum:$coursedom") {
- $output .= "$ca:$cd\n";
- } elsif ($line =~ m{^(.+)$}) {
- my @deps = split(/\s*,\s*/,$1);
- my @newdeps;
- my $changed = 0;
- foreach my $dep (@deps) {
- if ($dep =~ m{^/res/$coursedom/$coursenum/(.+)$}) {
- my $rest = $1;
- push(@newdeps,"/res/$cd/$ca/$rest");
- $checkdeps{$rest} = 1;
- $changed ++;
- } else {
- push(@newdeps,$dep);
- }
- }
- if ($changed) {
- $output .= ''.join(',',@newdeps).''."\n";
- }
- } else {
- $output .= "$line\n";
+ my ($needpriv,$needprivmeta);
+ if ($respublish) {
+ if ($getres) {
+ &Apache::lonnet::repcopy($docroot.$resurl.'/'.$file);
+ }
+ if ($getresmeta) {
+ &Apache::lonnet::repcopy($docroot.$resurl.'/'.$file.'.meta');
+ }
+ if (-e $docroot.$resurl.'/'.$file) {
+ if (&Apache::lonnet::repcopy_crsprivfile($srcurl.'/'.$file,$dest) eq 'ok') {
+ if (&Apache::londiff::are_different_files($docroot.$resurl.'/'.$file,$dest)) {
+ $needpriv = 1;
+ if (&File::Copy::copy($docroot.$resurl.'/'.$file,$dest)) {
+ if ($embstyle eq 'ssi') {
+ &crsres_fixup($dest,$coursenum,$coursedom,$ca,$cd);
}
}
- close($fh);
- if (open(my $fh,'>',$dest.'.meta')) {
- print $fh $output;
- close($fh);
+ } else {
+ if ($embstyle eq 'ssi') {
+ &crsres_fixup($dest,$coursenum,$coursedom,$ca,$cd,$subdir);
}
+ $newfile{$file} = $desturl.'/'.$subdir.'/'.$file;
}
}
+ } else {
+ $needpriv = 1;
}
- my ($ext) = ($file =~ /\.(\w+)$/);
- my $embstyle=&Apache::loncommon::fileembstyle($ext);
- if ($embstyle eq 'ssi') {
- my $outstring='';
- my $changes = 0;
- my @parser;
- $parser[0]=HTML::LCParser->new($src);
- $parser[-1]->xml_mode(1);
- my $token;
- while (@parser) {
- while ($token=$parser[-1]->get_token) {
- if ($token->[0] eq 'S') {
- my $tag=$token->[1];
- my $lctag=lc($tag);
- my %parms=%{$token->[2]};
- foreach my $type ('src','href','background','bgimg') {
- foreach my $key (keys(%parms)) {
- if ($key =~ /^$type$/i) {
- next if (($lctag eq 'img') && ($type eq 'src') &&
- ($parms{$key} =~ m{^data\:image/gif;base64,}));
- if ($parms{$key} =~ m{^\Q/res/$coursedom/$coursenum/\E}si) {
- $parms{$key} =~ s{^\Q/res/$coursedom/$coursenum/\E}{/res/$cd/$ca/$subdir/}si;
- $changes ++;
- }
- }
- }
- }
- # probably a image type