# The LearningOnline Network with CAPA
#
# $Id: imsimportdocs.pm,v 1.37 2018/05/02 17:06:45 raeburn Exp $
#
# Copyright Michigan State University Board of Trustees
#
# This file is part of the LearningOnline Network with CAPA (LON-CAPA).
#
# LON-CAPA is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# LON-CAPA is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with LON-CAPA; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# /home/httpd/html/adm/gpl.txt
#
# http://www.lon-capa.org/
#
package Apache::imsimportdocs;
use Apache::Constants qw(:common :http :methods);
use Apache::lonnet;
use Apache::londocs;
use Apache::loncommon;
use Apache::lonlocal;
use Apache::imsprocessor;
use LONCAPA::map();
use lib '/home/httpd/lib/perl/';
use LONCAPA;
use File::Path();
use strict;
sub jscript_one {
my %lt = &Apache::lonlocal::texthash(
se => 'Select',
to => 'Import topics only',
tp => 'Import topics + posts (with author)',
tn => 'Import topics + posts (no author)',
es => 'Enroll students only',
ea => 'Enroll all users',
nr => 'Not required',
id => 'You must select one of the additional options when importing Discussion Boards.',
ie => 'You must select one of the additional options when importing Enrollment.',
ct => 'You must check at least one Content Type.',
);
return <<"ENDJS";
function setOptions(caller,itemnum) {
var opForm = document.forms.pickoptions
var menu = 1 + itemnum*2
opForm.elements[menu].length = 0
if (opForm.elements[itemnum*2].checked == true) {
if (caller == "board") {
opForm.elements[menu].options[0] = new Option("$lt{'se'}","-1",true,true)
opForm.elements[menu].options[1] = new Option("$lt{'to'}","topics",true,true)
opForm.elements[menu].options[2] = new Option("$lt{'tp'}","allpost",true,true)
opForm.elements[menu].options[3] = new Option("$lt{'tn'}","allanon",true,true)
}
else {
if (caller == "users") {
opForm.elements[menu].length = 0
opForm.elements[menu].options[0] = new Option("$lt{'se'}","-1",true,true)
opForm.elements[menu].options[1] = new Option("$lt{'es'}","students",true,true)
opForm.elements[menu].options[2] = new Option("$lt{'ea'}","all",true,true)
}
}
}
else {
opForm.elements[menu].options[0] = new Option("$lt{'nr'}","0",true,true)
}
opForm.elements[menu].selectedIndex = 0
}
function verify(caller) {
var opForm = document.forms.pickoptions
var totcheck = 0;
for (var i=0; i<caller; i++) {
if (opForm.elements[2*i].checked == true) {
totcheck ++
if (opForm.elements[2*i].name == "board") {
if (opForm.elements[2*i+1].selectedIndex == 0) {
alert("$lt{'id'}")
return false
}
}
if (opForm.elements[2*i].name == "users") {
if (opForm.elements[2*i+1].selectedIndex == 0) {
alert("$lt{'ie'}")
return false
}
}
}
}
if (totcheck == 0) {
alert("$lt{'ct'}");
return false
}
return true
}
function nextPage(caller) {
if (verify(caller)) {
document.forms.pickoptions.submit()
}
}
ENDJS
}
sub jscript_two {
return <<"ENDJS";
function init(tf) {
setTimeout("self.close()",3000)
tf.submit();
}
ENDJS
}
sub handler {
my $r = shift;
&Apache::loncommon::content_type($r,'text/html');
$r->send_http_header;
return OK if $r->header_only;
my @areas = ();
my %cmsmap = ();
my %areaname = ();
&Apache::imsprocessor::ims_config(\@areas,\%cmsmap,\%areaname);
# get course data
my $coursenum=$env{'course.'.$env{'request.course.id'}.'.num'};
my $coursedom=$env{'course.'.$env{'request.course.id'}.'.domain'};
# get personal data
my $uname=$env{'user.name'};
my $udom=$env{'user.domain'};
my $plainname=&escape(
&Apache::loncommon::plainname($uname,$udom));
# does this user have privileges to post, etc?
my $allowed=&Apache::lonnet::allowed('mdc',$env{'request.course.id'});
unless ($allowed) {
$r->print(&Apache::loncommon::start_page('Import IMS package',undef,
{'only_body' => 1,}));
$r->print(
'<p class="LC_error">'
.&mt('Modification of Course Contents Disallowed')
.'</p><p>'
.&mt('Your current role does not grant you the right to modify course content in this course.')
.'</p>'
.&Apache::loncommon::end_page()
);
return OK;
}
my $javascript;
if ($env{'form.phase'} eq 'one') {
$javascript = &jscript_one();
} elsif ($env{'form.phase'} eq 'two') {
$javascript = &jscript_two();
}
$javascript =
"<script type=\"text/javascript\">\n".
"//<!--\n$javascript\n// --></script>\n";
my $headline = 'Import IMS package';
my $start_page = &Apache::loncommon::start_page($headline,
$javascript,
{'only_body' => 1,})
.'<h1>'.&mt($headline).'</h1>';
# print screen
$r->print($start_page);
if ($env{'form.phase'} eq 'one') {
&display_one($r,$coursenum,\@areas,\%areaname,%cmsmap);
} elsif ($env{'form.phase'} eq 'two') {
&display_two($r,$coursenum,$coursedom,$uname,$udom,\@areas,%cmsmap);
}
$r->print(&Apache::loncommon::end_page());
return OK;
}
sub display_one {
my ($r,$crs,$areasref,$areaname,%cmsmap) = @_;
my $cms = $env{'form.source'};
my $timenow = time;
my $tempdir = &Apache::imsprocessor::create_tempdir('DOCS',$crs,$timenow);
my $fname = &Apache::imsprocessor::uploadzip('DOCS',$tempdir);
my $unzip_result = '';
my $manifest_result = '';
unless ($tempdir eq '') {
$unzip_result = &Apache::imsprocessor::expand_zip($tempdir,$fname);
}
my %resources = ();
my %includedres = ();
my %includeditems = ();
my %items = ();
my %hrefs = ();
my %resinfo = ();
my %count = ();
my $counter = 0;
my %count = (
announce => 0,
board => 0,
doc => 0,
extlink => 0,
msg => 0,
pool => 0,
quiz => 0,
staff => 0,
survey => 0,
users => 0,
);
if ($unzip_result ne 'ok') {
$r->print(
&Apache::loncommon::confirmwrapper(
&Apache::lonhtmlcommon::confirm_success(
&mt('Processing of your IMS package failed because the file you'
.' uploaded could not be unzipped.'),1)
.'<br />'.&mt('Error: [_1]',$unzip_result))
);
return();
}
# Get manifest file from package
$manifest_result = &Apache::imsprocessor::process_manifest(
$cms,$tempdir,\%resources,\%items,\%hrefs,
\%resinfo,'choose',\%includedres,\%includeditems);
if ($manifest_result ne 'ok') {
$r->print(
'<br />'.&Apache::loncommon::confirmwrapper(
&Apache::lonhtmlcommon::confirm_success(
&mt('Unpacking of your IMS package failed because an IMS manifest file'
.' was not located in the package.'),1))
);
return();
}
# Count areas depending on cms version
foreach my $res (sort(keys(%resources))) {
if ($cms eq 'bb5' || $cms eq 'bb6' || $cms eq 'webctce4') {
foreach my $area (keys(%{$cmsmap{$cms}})) {
if ($resources{$res}{type} eq $cmsmap{$cms}{$area}) {
$count{$area} ++;
}
}
} elsif ($cms eq 'angel5') {
foreach my $area (keys(%{$cmsmap{$cms}})) {
if ($area eq 'doc') {
if (grep/^$resources{$res}{type}$/,@{$cmsmap{$cms}{doc}}) {
$count{$area} ++;
}
} elsif ($resources{$res}{type} eq $cmsmap{$cms}{$area}) {
$count{$area} ++;
}
}
} else { # Unknown cms format
$r->print(
'<span class="LC_warning">'
.&mt('Unsupported IMS format: [_1]',$cms)
.'</span><br />'
);
# return();
}
}
# Start output: Step 1 and step 2
$r->print(
'<form name="pickoptions" method="post" action="">'
.&Apache::lonhtmlcommon::topic_bar(
1,&mt('Choose which content types you wish to import'))
.'<p>'
.&mt('Check the checkboxes for all areas you wish to import from the IMS package:')
.'</p>'
);
$r->print(
&Apache::loncommon::start_data_table()
.&Apache::loncommon::start_data_table_header_row()
.'<th>'.&mt('Import?').'</th>'
.'<th>'.&mt('Content type').'</th>'
.'<th>'.&mt('Additional options').'</th>'
.&Apache::loncommon::end_data_table_header_row()
);
# Display import row for each area/content type
foreach my $area (@{$areasref}) {
unless ($count{$area} > 0) { next };
my $count_tag = 'flag_'.$counter;
# Checkbox: Import?
$r->print(
&Apache::loncommon::start_data_table_row()
.'<td><input name="'.$area.'" type="checkbox"'
);
if ($area eq 'board' || $area eq 'users') {
$r->print(qq| onclick='javascript:setOptions("$area","$counter")'|);
}
$r->print(' /></td>');
# Content Type
$r->print(
'<td>'
.$$areaname{$area}.' - '
.&mt('[quant,_1,item]',$count{$area})
.'</td>'
);
# Additional Options
$r->print('<td>');
if ($area eq 'board') {
$r->print(
'<select name="db_handling">'
.'<option value="-2"><-- '.&mt('Check Import first').'</option>'
.'</select>'
);
} elsif ($area eq 'users') {
$r->print(
'<select name="user_handling">'
.'<option value="-2"><-- '.&mt('Check Import first').'</option>'
.'</select>'
);
} else {
$r->print(
&mt('None')
.'<input type="hidden" name="'.$count_tag.'" />'
);
}
$r->print('</td>');
$r->print(&Apache::loncommon::end_data_table_row());
$counter ++;
}
$r->print(&Apache::loncommon::end_data_table());
$r->print(
&Apache::lonhtmlcommon::topic_bar(
2,&mt('Choose display options for listing of contents of top level of package'))
.'<p>'
.&mt('Select a display option for the package content:')
.'</p>'
);
$r->print(
'<label>'
.'<input type="radio" name="toplevel" value="newfolder" />'
.&mt('Display listing of contents in a new folder, with folder name:')
.'</label>'
.' <input type="text" name="foldername" size="15" value="'.&mt('Type Name Here').'" />'
.'<br />'
.'<label>'
.'<input type="radio" name="toplevel" value="oldfolder" checked="checked" />'
.&mt('Append listing of contents of top level of package to contents list for the current folder.')
.'</label>'
);
# Buttons
$r->print(
'<input type="hidden" name="folder" value="'.$env{'form.folder'}.'" />'
.'<input type="hidden" name="source" value="'.$cms.'" />'
.'<input type="hidden" name="tempdir" value="'.$tempdir.'" />'
.'<input type="hidden" name="phase" value="two" />'
);
$r->print(
'<hr />'
.'<p>'
.'<input type="button" name="exitpage" value="'.&mt('Cancel').'"'
.' onclick="javascript:self.close()" />'
.' '
.'<input type="button" name="nextpage" value="'.&mt('Finish Import').'"'
.' onclick="javascript:nextPage('.$counter.')" />'
.'</p>'
);
$r->print('</form>');
}
sub display_two {
my ($r,$crs,$cdom,$uname,$udom,$areas,%cmsmap) = @_;
my $folder = $env{'form.folder'};
my $cms = $env{'form.source'};
my $tempdir = $env{'form.tempdir'};
my %importareas = ();
my %includedres = ();
my %includeditems = ();
my @targets = ();
my %resources = ();
my %items = ();
my %hrefs = ();
my %urls = ();
my %resinfo = ();
my %total = (
page => 0,
prob => 0,
seq => 0,
board => 0,
quiz => 0,
surv => 0,
);
my @pages = ();
my @sequences = ();
my @resrcfiles = ();
my $timenow = time;
my $destdir = $Apache::lonnet::perlvar{'lonDocRoot'}.'/userfiles/'.$cdom.'/'.$crs.'/'.$timenow;
my $seqstem = "/uploaded/$cdom/$crs/$timenow";
my $db_handling = '';
my $user_handling = '';
my $toplevel = '';
my $foldername = '';
my %topitems = ();
if (defined($env{'form.toplevel'}) ) {
$toplevel = $env{'form.toplevel'};
}
if (defined($env{'form.foldername'}) ) {
$foldername = $env{'form.foldername'};
}
foreach my $area (@{$areas}) {
if (defined($env{"form.$area"}) && ($env{'form.'.$area} ne '')) {
if ($cms eq 'angel5' && $area eq 'doc') {
foreach (@{$cmsmap{$cms}{$area}}) {
$importareas{$_} = 1;
}
} else {
$importareas{$cmsmap{$cms}{$area}} = 1;
}
if ($area eq 'board') {
$db_handling = $env{'form.db_handling'};
} elsif ($area eq 'users') {
$user_handling = $env{'form.user_handling'};
}
}
}
my $manifest_result = &Apache::imsprocessor::process_manifest($cms,$tempdir,\%resources,\%items,\%hrefs,\%resinfo,'prepare',\%includedres,\%includeditems);
if ($manifest_result eq 'ok') {
foreach my $res (sort(keys(%resources))) {
if ($importareas{$resources{$res}{type}}) {
$includedres{$res} = 1;
}
}
foreach my $itm (sort(keys(%items))) {
&Apache::imsprocessor::get_imports(\%includeditems,\%items,\%resources,\%importareas,$itm);
}
}
foreach my $itm (sort(keys(%includeditems))) {
&Apache::imsprocessor::get_parents(\%includeditems,\%items,$itm);
}
$manifest_result = &Apache::imsprocessor::process_manifest($cms,$tempdir,\%resources,\%items,\%hrefs,\%resinfo,'build',\%includedres,\%includeditems);
if ($manifest_result eq 'ok') {
my @path = ($cdom,$crs,$timenow);
my $fullpath = $Apache::lonnet::perlvar{'lonDocRoot'}.'/userfiles';
foreach my $item (@path) {
$fullpath .= '/'.$item;
if (!-e "$fullpath") {
mkdir("$fullpath",0770);
}
}
my @namedirs = ("resfiles","sequences","pages","problems");
foreach my $name (@namedirs) {
if (!-e "$fullpath/$name") {
mkdir("$fullpath/$name",0770);
}
}
&Apache::imsprocessor::target_resources(\%resources,\%importareas,\@targets);
my @boards = ();
my @announcements = ();
my @quizzes = ();
my @surveys = ();
my @pools = ();
my @groups = ();
my %messages = ();
my @timestamp = ();
my %boardnum = ();
my @topurls = ();
my @topnames = ();
my @packages = ();
&Apache::imsprocessor::process_resinfo($cms,'DOCS',$tempdir,$destdir,\%items,\%resources,\@targets,\@boards,\@announcements,\@quizzes,\@surveys,\@pools,\@groups,\%messages,\@timestamp,\%boardnum,\%resinfo,$udom,$uname,$cdom,$crs,$db_handling,$user_handling,\%total,$seqstem,$seqstem,\@resrcfiles,\@packages,\%hrefs,\@pages,\@sequences);
my $copy_result = &Apache::imsprocessor::copy_resources('DOCS',$cms,\%hrefs,\%resources,$tempdir,\@targets,\%urls,$crs,$cdom,$destdir,$timenow,\%importareas);
&Apache::imsprocessor::build_structure($cms,'DOCS',$destdir,\%items,\%resinfo,\%resources,\@targets,\%hrefs,$udom,$uname,'',$timenow,$cdom,$crs,\@timestamp,\%total,\@boards,\@announcements,\@quizzes,\@surveys,\@pools,\%boardnum,\@pages,\@sequences,\@topurls,\@topnames,\@packages,\%includeditems);
foreach my $item (@pages) {
my $filename = $timenow.'/pages/'.$item;
my $fetchresult= &Apache::lonnet::process_coursefile('propagate',$crs,$cdom,$filename,'');
}
foreach my $item (@sequences) {
unless ($item eq 'Top.sequence' && $toplevel eq 'oldfolder') {
my $filename = $timenow.'/sequences/'.$item;
my $fetchresult= &Apache::lonnet::process_coursefile('propagate',$crs,$cdom,$filename,'');
}
}
foreach my $item (@resrcfiles) {
my $filename = $timenow.'/resfiles/'.$item;
my $fetchresult= &Apache::lonnet::process_coursefile('propagate',$crs,$cdom,$filename,'');
}
my @imports = ();
if ($toplevel eq 'oldfolder') {
for (my $i=0; $i<@topurls; $i++) {
my $url = &unescape($topurls[$i]);
my $name = &unescape($topnames[$i]);
push(@imports, [$name, $url]);
}
} elsif ($toplevel eq 'newfolder') {
my $url = &unescape("/uploaded/$cdom/$crs/$timenow/sequences/Top.sequence");
my $name = &unescape("$env{'form.foldername'}");
push(@imports, [$name, $url]);
}
my $errtext='';
my $fatal=0;
($errtext,$fatal)= &Apache::londocs::mapread($crs,$cdom,$folder.'.sequence');
if ($#LONCAPA::map::order<1) {
$LONCAPA::map::order[0]=1;
$LONCAPA::map::resources[1]='';
}
my ($errtext,$fatal)=&Apache::londocs::group_import($crs,$cdom,$folder,'sequence','imsimport',@imports);
if ($fatal) {
&Apache::lonnet::logthis("Fatal error during group_import.");
}
}
if ($tempdir =~ m/^\/home\/httpd\/perl\/tmp\/$crs\/\d{10}/) {
&File::Path::remove_tree($tempdir,{ safe => 1 });
}
# All done, display success message
$r->print(
'<p class="LC_success">'
.&mt('Your import is complete.')
.'</p>'
);
# Re-initialize Button
my $initbutton =
'<input type="button" value="'
.&mt('re-initializing Course')
.'" onclick="javascript:init(this.form)" />';
my $windowname = 'loncapaclient';
if ($env{'request.lti.login'}) {
$windowname .= 'lti';
}
$r->print(
'<form method="post" action="/adm/roles" target="'.$windowname.'" name="importDone">'
.'<input type="hidden" name="orgurl" value="/adm/coursedocs" />'
.'<input type="hidden" name="selectrole" value="1" />'
.'<input type="hidden" name="'.$env{'request.role'}.'" value="1" />'
.'<p class="LC_warning">'
.&mt('Changes will become active for your current session after [_1]'
.', or the next time you log in.'
,$initbutton)
.'</p>'
.'</form>'
);
}
1;
__END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>