--- loncom/imspackages/imsprocessor.pm 2005/03/23 20:55:33 1.18
+++ loncom/imspackages/imsprocessor.pm 2005/10/25 13:54:46 1.30
@@ -24,6 +24,9 @@
package Apache::imsprocessor;
use Apache::lonnet;
+use Apache::loncleanup;
+use LWP::UserAgent;
+use HTTP::Request::Common;
use LONCAPA::Configuration;
use strict;
@@ -41,7 +44,17 @@ sub ims_config {
survey => 'assessment/x-bb-survey',
users => 'course/x-bb-user',
);
- %{$$cmsmap{bb6}} = %{$$cmsmap{bb5}};
+ %{$$cmsmap{bb6}} = (
+ announce => 'resource/x-bb-announcement',
+ board => 'resource/x-bb-discussionboard',
+ doc => 'resource/x-bb-document',
+ extlink => 'resource/x-bb-externallink',
+ pool => 'assessment/x-bb-qti-pool',
+ quiz => 'assessment/x-bb-qti-test',
+ staff => 'resource/x-bb-staffinfo',
+ survey => 'assessment/x-bb-survey',
+ users => 'course/x-bb-user',
+ );
$$cmsmap{bb6}{conference} = 'resource/x-bb-conference';
%{$$cmsmap{angel}} = (
board => 'BOARD',
@@ -95,7 +108,7 @@ sub uploadzip {
my ($context,$tempdir,$source) = @_;
my $fname;
if ($context eq 'DOCS') {
- $fname=$ENV{'form.uploadname.filename'};
+ $fname=$env{'form.uploadname.filename'};
# Replace Windows backslashes by forward slashes
$fname=~s/\\/\//g;
# Get rid of everything but the actual filename
@@ -107,9 +120,9 @@ sub uploadzip {
# See if there is anything left
unless ($fname) { return 'error: no uploaded file'; }
# Save the file
- chomp($ENV{'form.uploadname'});
+ chomp($env{'form.uploadname'});
open(my $fh,'>'.$tempdir.'/'.$fname);
- print $fh $ENV{'form.uploadname'};
+ print $fh $env{'form.uploadname'};
close($fh);
} elsif ($context eq 'CSTR') {
if ($source =~ m/\/([^\/]+)$/) {
@@ -128,14 +141,8 @@ sub expand_zip {
return 'no zip';
}
if ($filename =~ m|\.zip$|i) {
- # unzip can cause an sh launch which can pass along all of %ENV
- # which can be too large for /bin/sh to handle
- my %oldENV=%ENV;
- undef(%ENV);
open(OUTPUT, "unzip -o $zipfile -d $tempdir 2> /dev/null |");
close(OUTPUT);
- %ENV=%oldENV;
- undef(%oldENV);
} else {
return 'nozip';
}
@@ -517,15 +524,15 @@ sub process_resinfo {
$board_id ++;
$board_count ++;
}
- } elsif ($$resources{$key}{type} eq "assessment/x-bb-pool") {
+ } elsif ($$resources{$key}{type} =~/assessment\/x\-bb\-(qti\-)?pool/) {
%{$$resinfo{$key}} = ();
&process_assessment($cms,$context,$key,$docroot,'pool',$dirname,$destdir,\%{$$resinfo{$key}},$total,$udom,$uname,$pagesfiles,$sequencesfiles,$randompicks,\$dbparse,$resources,$items,\%catinfo,\%qzdbsettings,$hrefs);
push @{$pools}, $key;
- } elsif ($$resources{$key}{type} eq "assessment/x-bb-quiz") {
+ } elsif ($$resources{$key}{type} =~ /assessment\/x\-bb\-(qti\-)?quiz/) {
%{$$resinfo{$key}} = ();
&process_assessment($cms,$context,$key,$docroot,'quiz',$dirname,$destdir,\%{$$resinfo{$key}},$total,$udom,$uname,$pagesfiles,$sequencesfiles,$randompicks,\$dbparse,$resources,$items,\%catinfo,\%qzdbsettings,$hrefs);
push @{$quizzes}, $key;
- } elsif ($$resources{$key}{type} eq "assessment/x-bb-survey") {
+ } elsif ($$resources{$key}{type} =~ /assessment\/x\-bb\-(qti\-)?survey/) {
%{$$resinfo{$key}} = ();
&process_assessment($cms,$context,$key,$docroot,'survey',$dirname,$destdir,\%{$$resinfo{$key}},$total,$udom,$uname,$pagesfiles,$sequencesfiles,$randompicks,\$dbparse,$resources,$items,\%catinfo,\%qzdbsettings,$hrefs);
push @{$surveys}, $key;
@@ -1730,7 +1737,7 @@ sub parse_bb5_assessment {
$id = $attr->{id};
} elsif ( ($state[0] eq $toptag{$container}) && ($state[1] =~ m/^QUESTION_(\w+)$/) && ($state[2] eq "BODY") && ($state[3] eq "FLAGS") ) {
if ($state[4] eq "ISHTML") {
- $$settings{$id}{html} = $attr->{value};
+ $$settings{$id}{ishtml} = $attr->{value};
} elsif ($state[4] eq "ISNEWLINELITERAL") {
$$settings{$id}{newline} = $attr->{value};
}
@@ -1830,8 +1837,238 @@ sub parse_bb5_assessment {
}
sub parse_bb6_assessment {
- my ($res,$docroot,$container,$settings,$allanswers,$allchoices,$allids) = @_;
- return;
+ my ($res,$docroot,$container,$settings,$allids) = @_;
+ my $xmlfile = $docroot.'/'.$res.".dat";
+ my @state = ();
+ my $id; # the current question ID
+ my $response; # the current response ID
+ my $foil; # the current foil ID
+ my $numchoice; # the current right match choice;
+ my $labelcount; # the current count of choices for a matching item.
+ my $curr_shuffle;
+ my $curr_class; # the current question type
+ my $curr_matchitem;
+ my $curr_block_type; # the current block type
+ my $curr_flow; # the current flow class attribute
+ my $curr_flow_mat; # the current flow_mat class attribute
+ my $curr_feedback_type; # the current feedback type
+ my $numorder; # counter for ordering type questions
+
+ my $itemfrag = "questestinterop assessment section item";
+ my $presfrag = "$itemfrag presentation flow flow";
+ my $blockflow = 'flow';
+ my $responselid;
+ my $instructionfrag = "questestinterop assessment presentation_material flow_mat material";
+ my $feedbackfrag = "$itemfrag itemfeedback";
+ my $feedback_tag = '';
+ my $responselid;
+ my $p = HTML::Parser->new
+ (
+ xml_mode => 1,
+ start_h =>
+ [sub {
+ my ($tagname, $attr) = @_;
+ push @state, $tagname;
+ if ("@state" eq "questestinterop assessment") {
+ $$settings{title} = $attr->{title};
+ }
+ if ("@state" eq "questestinterop assessment rubric flow_mat material mat_extension mat_formattedtext") {
+ $$settings{description}{texttype} = $attr->{type};
+ }
+ if ("@state" eq $presfrag) {
+ if ($attr->{class} eq 'QUESTION_BLOCK') {
+ $curr_block_type = 'question';
+ } elsif ($attr->{class} eq 'RESPONSE_BLOCK') {
+ $curr_block_type = 'response';
+ if ($curr_class eq 'Matching') {
+ $responselid = 'flow response_lid';
+ } else {
+ $responselid = 'response_lid';
+ }
+ } elsif (($attr->{class} eq 'RIGHT_MATCH_BLOCK')) {
+ $numchoice = 0;
+ $curr_block_type = 'rightmatch';
+ }
+ }
+ if ("@state" eq "$presfrag flow") {
+ if (($curr_block_type =~ /^rightmatch/) && ($attr->{class} eq 'Block')) {
+ $curr_block_type = 'rightmatch'.$numchoice;
+ $numchoice ++;
+ }
+ }
+ if ($state[-1] eq 'flow') {
+ $curr_flow = $attr->{class};
+ }
+ if ($state[-1] eq 'flow_mat') {
+ $curr_flow_mat = $attr->{class};
+ }
+ if ("@state" eq "$presfrag $blockflow material mat_extension mat_formattedtext") {
+ $$settings{$id}{$curr_block_type}{texttype} = $attr->{texttype};
+ }
+ if ("@state" eq "$presfrag $blockflow material matapplication") {
+ $$settings{$id}{$curr_block_type}{image} = $attr->{uri};
+ $$settings{$id}{$curr_block_type}{style} = $attr->{embedded};
+ $$settings{$id}{$curr_block_type}{label} = $attr->{label};
+ }
+ if ("@state" eq "$presfrag $blockflow material mattext") {
+ $$settings{$id}{$curr_block_type}{link} = $attr->{uri};
+ }
+ if ("@state" eq "$presfrag $responselid") {
+ $response = $attr->{ident};
+ $labelcount = 0;
+ if ($curr_class eq 'Matching') {
+ push(@{$$settings{$id}{answers}},$response);
+ %{$$settings{$id}{$response}} = ();
+ foreach my $key (keys(%{$$settings{$id}{$curr_block_type}})) {
+ $$settings{$id}{$response}{$key} = $$settings{$id}{$curr_block_type}{$key};
+ }
+ %{$$settings{$id}{$curr_block_type}} = ();
+ }
+ }
+ if ("@state" eq "$presfrag $responselid render_choice") {
+ $curr_shuffle = $attr->{shuffle};
+ }
+ if ("@state" eq "$presfrag $responselid render_choice flow_label response_label") {
+ $foil = $attr->{ident};
+ %{$$settings{$id}{$foil}} = ();
+ $$settings{$id}{$foil}{randomize} = $curr_shuffle;
+ unless ($curr_class eq 'Essay'){
+ if ($curr_class eq 'Matching') {
+ push(@{$$settings{$id}{$response}{items}},$foil);
+ $$settings{$id}{$foil}{order} = $labelcount;
+ $labelcount ++;
+ } else {
+ push(@{$$settings{$id}{answers}},$foil);
+ @{$$settings{$id}{correctanswer}} = ();
+ }
+ }
+ }
+ if ("@state" eq "$presfrag $responselid render_choice flow_label response_label flow_mat material matapplication") {
+ $$settings{$id}{$foil}{filetype} = $attr->{embedded};
+ $$settings{$id}{$foil}{label} = $attr->{label};
+ $$settings{$id}{$foil}{uri} = $attr->{uri};
+ }
+ if ("@state" eq "$presfrag $responselid render_choice flow_label response_label flow_mat material mattext") {
+ $$settings{$id}{$foil}{link} = $attr->{uri};
+ }
+ if ("@state" eq "questestinterop assessment section item resprocessing") {
+ if ($curr_class eq 'Matching') {
+ $$settings{$id}{allchoices} = $numchoice;
+ }
+ }
+ if ("@state" eq "questestinterop assessment section item resprocessing respcondition conditionvar varequal") {
+ if ($curr_class eq 'Matching') {
+ $curr_matchitem = $attr->{respident};
+ }
+ }
+ if ("@state" eq $feedbackfrag) {
+ $curr_feedback_type = $attr->{ident};
+ $feedback_tag = "";
+ }
+ if ("@state" eq "$feedbackfrag solution") {
+ $curr_feedback_type = 'solution';
+ $feedback_tag = "solution solutionmaterial";
+ }
+ if ("@state" eq "$feedbackfrag $feedback_tag flow_mat flow_mat material matapplication") {
+ $$settings{$id}{$curr_feedback_type.'feedback'}{filetype} = $attr->{'embedded'};
+ $$settings{$id}{$curr_feedback_type.'feedback'}{label} = $attr->{label};
+ $$settings{$id}{$curr_feedback_type.'feedback'}{uri} = $attr->{uri};
+ }
+ if ("@state" eq "$feedbackfrag $feedback_tag flow_mat flow_mat material mattext") {
+ $$settings{$id}{$curr_feedback_type.'feedback'}{link} = $attr->{uri};
+ }
+ }, "tagname, attr"],
+ text_h =>
+ [sub {
+ my ($text) = @_;
+ $text =~ s/^\s+//g;
+ $text =~ s/\s+$//g;
+ if ("@state" eq "questestinterop assessment rubric flow_mat material mat_extension mat_formattedtext") {
+ $$settings{description}{text} = $text;
+ }
+ if ("@state" eq "questestinterop assessment rubric flow_mat material mattext") {
+ $$settings{description}{text} = $text;
+ }
+ if ("@state" eq "$instructionfrag mat_extension mat_formattedtext") {
+ $$settings{instructions}{text} = $text;
+ }
+ if ("@state" eq "$instructionfrag mattext") {
+ $$settings{instructions}{text} = $text;
+ }
+ if ("@state" eq "questestinterop assessment section item itemmetadata bbmd_asi_object_id") {
+ $id = $text;
+ push @{$allids}, $id;
+ %{$$settings{$id}} = ();
+ @{$$settings{$id}{answers}} = ();
+ %{$$settings{$id}{question}} = ();
+ %{$$settings{$id}{correctfeedback}} = ();
+ %{$$settings{$id}{incorrectfeedback}} = ();
+ %{$$settings{$id}{solutionfeedback}} = ();
+ }
+ if ("@state" eq "questestinterop assessment section item itemmetadata bbmd_questiontype") {
+ $$settings{$id}{class} = $text;
+ $curr_class = $text;
+ if ($curr_class eq 'Matching') {
+ $blockflow = 'flow flow';
+ } else {
+ $blockflow = 'flow';
+ }
+ }
+ if ("@state" eq "$presfrag $blockflow material mat_extension mat_formattedtext") {
+ $$settings{$id}{$curr_block_type}{text} = $text;
+ }
+ if ("@state" eq "$presfrag $blockflow material mattext") {
+ if ($curr_flow eq 'LINK_BLOCK') {
+ $$settings{$id}{$curr_block_type}{linkname} = $text;
+ } elsif ($curr_flow eq 'FORMATTED_TEXT_BLOCK') {
+ $$settings{$id}{$curr_block_type}{text} = $text;
+ }
+ }
+ if ("@state" eq "$presfrag $responselid render_choice flow_label response_label flow_mat material mat_extension mat_formattedtext") {
+ $$settings{$id}{$foil}{text} = $text;
+ }
+ if ("@state" eq "$presfrag $responselid render_choice flow_label response_label flow_mat material mattext") {
+ if ($curr_flow_mat eq 'LINK_BLOCK') {
+ $$settings{$id}{$foil}{linkname} = $text;
+ } else {
+ $$settings{$id}{$foil}{text} = $text;
+ }
+ }
+ if ("@state" eq "questestinterop assessment section item resprocessing respcondition conditionvar varequal") {
+ if ($curr_class eq 'Matching') {
+ $$settings{$id}{$curr_matchitem}{correctanswer} = $text;
+ } else {
+ push(@{$$settings{$id}{correctanswer}},$text);
+ }
+ }
+ if ("@state" eq "questestinterop assessment section item resprocessing respcondition conditionvar") {
+ $numorder = 0;
+ }
+ if ("@state" eq "questestinterop assessment section item resprocessing respcondition conditionvar and varequal") {
+ push(@{$$settings{$id}{correctanswer}},$text);
+ if ($curr_class eq 'Ordering') {
+ $numorder ++;
+ $$settings{$id}{$text}{order} = $numorder;
+ }
+ }
+ if ("@state" eq "$feedbackfrag $feedback_tag flow_mat flow_mat material mat_extension mat_formattedtext") {
+ $$settings{$id}{$curr_feedback_type.'feedback'}{text} = $text;
+ }
+ if ("@state" eq "$feedbackfrag $feedback_tag flow_mat flow_mat material mattext") {
+ $$settings{$id}{$curr_feedback_type.'feedback'}{linkname} = $text;
+ }
+ }, "dtext"],
+ end_h =>
+ [sub {
+ my ($tagname) = @_;
+ pop @state;
+ }, "tagname"],
+ );
+ $p->unbroken_text(1);
+ $p->marked_sections(1);
+ $p->parse_file($xmlfile);
+ $p->eof;
+ return;
}
sub parse_webct4_assessment {
@@ -2098,7 +2335,7 @@ sub parse_webct4_questionDB {
$$settings{$id}{$numid}{toltype} = $attr->{type};
}
if ("@state" eq "questestinterop section item resprocessing itemproc_extension webct:x_webct_v01_autocalculate webct:x_webct_v01_unit") {
- my $unitid = $attr->{ident};
+ $unitid = $attr->{ident};
%{$$settings{$id}{$numid}{$unitid}} = ();
push(@{$$settings{$id}{$numid}{units}},$unitid);
$$settings{$id}{$numid}{$unitid}{value} = $attr->{value};
@@ -2290,7 +2527,7 @@ sub process_assessment {
my $randompickflag = 0;
my ($cid,$cdom,$cnum);
if ($context eq 'DOCS') {
- $cid = $ENV{'request.course.id'};
+ $cid = $env{'request.course.id'};
($cdom,$cnum) = split/_/,$cid;
}
my $destresdir = $destdir;
@@ -2302,7 +2539,7 @@ sub process_assessment {
if ($cms eq 'bb5') {
&parse_bb5_assessment($res,$docroot,$container,$settings,\%allanswers,\%allchoices,\@allids);
} elsif ($cms eq 'bb6') {
- &parse_bb6_assessment($res,$docroot,$container,$settings,\%allanswers,\%allchoices,\@allids);
+ &parse_bb6_assessment($res,$docroot,$container,$settings,\@allids);
} elsif ($cms eq 'webct4') {
unless($$dbparse) {
&parse_webct4_questionDB($docroot,$$resources{$res}{file},$catinfo,$qzdbsettings,\%alldbanswers,\%alldbchoices,\@alldbquestids);
@@ -2348,7 +2585,7 @@ sub process_assessment {
if (!-e "$destdir/problems/$seqname") {
mkdir("$destdir/problems/$seqname",0755);
}
- my $newdir = "$destdir/problems/$seqname";
+ $newdir = "$destdir/problems/$seqname";
my $dbcontainerdir;
&build_problem_container($cms,$seqname,$destdir,'database',$seqname,$total,$sequencesfiles,$pagesfiles,$randompickflag,$context,\@{$$catinfo{$category}{contents}},$udom,$uname,$dirname,\$dbcontainerdir,$cid,$cdom,$cnum,$catinfo,$qzdbsettings);
}
@@ -2376,14 +2613,18 @@ sub process_assessment {
if (!-e "$destdir/problems/$dirtitle") {
mkdir("$destdir/problems/$dirtitle",0755);
}
- my $newdir = "$destdir/problems/$dirtitle";
+ $newdir = "$destdir/problems/$dirtitle";
}
- &build_problem_container($cms,$dirtitle,$destdir,$container,$res,$total,$sequencesfiles,$pagesfiles,$randompickflag,$context,\@allids,$udom,$uname,$dirname,\$containerdir,$cid,$cdom,$cnum,$catinfo,$qzdbsettings);
+ if ($cms eq 'webct4') {
+ &build_problem_container($cms,$dirtitle,$destdir,$container,$res,$total,$sequencesfiles,$pagesfiles,$randompickflag,$context,\@allids,$udom,$uname,$dirname,\$containerdir,$cid,$cdom,$cnum,$catinfo,$qzdbsettings);
+ } else {
+ &build_problem_container($cms,$dirtitle,$destdir,$container,$res,$total,$sequencesfiles,$pagesfiles,$randompickflag,$context,\@allids,$udom,$uname,$dirname,\$containerdir,$cid,$cdom,$cnum,$catinfo,$settings);
+ }
if ($cms eq 'bb5') {
- &write_bb5_questions(\@allids,$containerdir,$context,$settings,$dirname,$res,\%allanswers,\%allchoices,$total,$newdir,$cid,$cdom,$cnum);
+ &write_bb5_questions(\@allids,$containerdir,$context,$settings,$dirname,$destdir,$res,\%allanswers,\%allchoices,$total,$newdir,$cid,$cdom,$cnum,$docroot);
} elsif ($cms eq 'bb6') {
- &write_bb6_questions(\@allids,$containerdir,$context,$settings,$dirname,$res,\%allanswers,\%allchoices,$total,$newdir,$cid,$cdom,$cnum);
+ &write_bb6_questions(\@allids,$containerdir,$context,$settings,$dirname,$destdir,$res,$total,$newdir,$cid,$cdom,$cnum,$docroot);
}
}
@@ -2417,7 +2658,11 @@ sub build_problem_container {
my $probsrc = "/res/lib/templates/simpleproblem.problem";
if ($context eq 'CSTR') {
foreach my $id (@{$allids}) {
- $probtitle{$id} = $$settings{$id}{title};
+ if ($cms eq 'webct4') {
+ $probtitle{$id} = $$settings{$id}{title};
+ } else {
+ $probtitle{$id} = $$settings{title};
+ }
$probtitle{$id} =~ s/\s/_/g;
$probtitle{$id} =~ s/\W//g;
$probtitle{$id} .= '_'.$id;
@@ -2429,7 +2674,7 @@ sub build_problem_container {
$probdir =~ s/\W//g;
$probsrc = "$dirname/problems/$probdir/$probtitle{$$allids[0]}.problem";
} else {
- $probsrc="$dirname/problems/$dirtitle/$$allids[0].problem";
+ $probsrc="$dirname/problems/$dirtitle/$probtitle{$$allids[0]}.problem";
}
}
print $fh qq|
#
#g;
$qnum ++;
my $output;
my $permcontainer = $containerdir;
@@ -2817,7 +3072,11 @@ sub write_bb5_questions {
if ($context eq 'CSTR') {
$output .= qq|
|;
- open(PROB,">$newdir/problems/$id.problem");
+ my $title = $$settings{title};
+ $title =~ s/\s/_/g;
+ $title =~ s/\W//g;
+ $title .= '_'.$id;
+ open(PROB,">:utf8", "$newdir/$title.problem");
print PROB $output;
close PROB;
} else {
@@ -2845,15 +3104,12 @@ sub write_webct4_questions {
$allfeedback .= $feedback;
}
if ($$settings{$id}{texttype} eq 'text/html') {
- $$settings{$id}{text} = &HTML::Entities::decode($$settings{$id}{text});
- $$settings{$id}{text} = &Apache::lonxml::htmlclean($$settings{$id}{text});
- $$settings{$id}{text} =~ s#(]+?)(/?>)#$1../../resfiles/$2 />#gi;
+ $$settings{$id}{text} =~ s#<([bh])r>#<$1r />#g;
$$settings{$id}{text} =~ s#
#
#g;
- $$settings{$id}{text} =~ s#<\\p>##g;
+ $$settings{$id}{text} =~ s#