--- loncom/homework/grades.pm 2019/02/23 17:34:59 1.596.2.12.2.41.2.6 +++ loncom/homework/grades.pm 2018/09/11 14:35:56 1.596.2.12.2.43 @@ -1,7 +1,7 @@ # The LearningOnline Network with CAPA # The LON-CAPA Grading handler # -# $Id: grades.pm,v 1.596.2.12.2.41.2.6 2019/02/23 17:34:59 raeburn Exp $ +# $Id: grades.pm,v 1.596.2.12.2.43 2018/09/11 14:35:56 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -44,8 +44,6 @@ use Apache::Constants qw(:common :http); use Apache::lonlocal; use Apache::lonenc; use Apache::bridgetask(); -use HTML::Parser(); -use File::MMagic; use String::Similarity; use LONCAPA; @@ -4216,7 +4214,7 @@ sub editgrades { my ($uname,$udom)=split(/:/,$user); my %newrecord; my $updateflag = 0; - my $usec=$classlist->{"$uname:$udom"}[5]; + my $usec=$classlist->{"$uname:$udom"}[5]; my $canmodify = &canmodify($usec); my $line = ''. &nameUserString(undef,$$fullname{$user},$uname,$udom).''; @@ -4609,8 +4607,8 @@ sub csvuploadmap { $datatoken=&Apache::loncommon::upfile_store($request); } else { $datatoken=&Apache::loncommon::valid_datatoken($env{'form.datatoken'}); - if ($datatoken ne '') { - &Apache::loncommon::load_tmp_file($request,$datatoken); + if ($datatoken ne '') { + &Apache::loncommon::load_tmp_file($request,$datatoken); } } my @records=&Apache::loncommon::upfile_record_sep(); @@ -5718,7 +5716,7 @@ sub scantron_uploads { sub scantron_scantab { my $result=' - '.&Apache::loncommon::start_data_table('LC_scantron_action').' - '.&Apache::loncommon::start_data_table_header_row().' - -  '.&mt('Specify a bubblesheet data file to upload.').' - - '.&Apache::loncommon::end_data_table_header_row().' - '.&Apache::loncommon::start_data_table_row().' - - '.&mt('File to upload: [_1]','').'
'."\n"); - if ($formatoptions) { - $r->print(' - '.&Apache::loncommon::end_data_table_row().' - '.&Apache::loncommon::start_data_table_row().' - '.$formattitle.(' 'x2).$formatoptions.' - - '.&Apache::loncommon::end_data_table_row().' - '.&Apache::loncommon::start_data_table_row().' - ' - ); - } else { - $r->print('
'); - } - $r->print(' - - '.&Apache::loncommon::end_data_table_row().' - '.&Apache::loncommon::end_data_table().' - ' - ); + '.&mt('File to upload: [_1]','').' +
+ + +'); + $r->print(' + + '.&Apache::loncommon::end_data_table_row().' + '.&Apache::loncommon::end_data_table().' +'); } # Chunk of form to prompt for a file to grade and how: @@ -5976,6 +6023,98 @@ sub scantron_selectphase { return; } +=pod + +=item get_scantron_config + + Parse and return the scantron configuration line selected as a + hash of configuration file fields. + + Arguments: + which - the name of the configuration to parse from the file. + + + Returns: + If the named configuration is not in the file, an empty + hash is returned. + a hash with the fields + name - internal name for the this configuration setup + description - text to display to operator that describes this config + CODElocation - if 0 or the string 'none' + - no CODE exists for this config + if -1 || the string 'letter' + - a CODE exists for this config and is + a string of letters + Unsupported value (but planned for future support) + if a positive integer + - The CODE exists as the first n items from + the question section of the form + if the string 'number' + - The CODE exists for this config and is + a string of numbers + CODEstart - (only matter if a CODE exists) column in the line where + the CODE starts + CODElength - length of the CODE + IDstart - column where the student/employee ID starts + IDlength - length of the student/employee ID info + Qstart - column where the information from the bubbled + 'questions' start + Qlength - number of columns comprising a single bubble line from + the sheet. (usually either 1 or 10) + Qon - either a single character representing the character used + to signal a bubble was chosen in the positional setup, or + the string 'letter' if the letter of the chosen bubble is + in the final, or 'number' if a number representing the + chosen bubble is in the file (1->A 0->J) + Qoff - the character used to represent that a bubble was + left blank + PaperID - if the scanning process generates a unique number for each + sheet scanned the column that this ID number starts in + PaperIDlength - number of columns that comprise the unique ID number + for the sheet of paper + FirstName - column that the first name starts in + FirstNameLength - number of columns that the first name spans + + LastName - column that the last name starts in + LastNameLength - number of columns that the last name spans + BubblesPerRow - number of bubbles available in each row used to + bubble an answer. (If not specified, 10 assumed). + +=cut + +sub get_scantron_config { + my ($which) = @_; + my @lines = &get_scantronformat_file(); + my %config; + #FIXME probably should move to XML it has already gotten a bit much now + foreach my $line (@lines) { + my ($name,$descrip)=split(/:/,$line); + if ($name ne $which ) { next; } + chomp($line); + my @config=split(/:/,$line); + $config{'name'}=$config[0]; + $config{'description'}=$config[1]; + $config{'CODElocation'}=$config[2]; + $config{'CODEstart'}=$config[3]; + $config{'CODElength'}=$config[4]; + $config{'IDstart'}=$config[5]; + $config{'IDlength'}=$config[6]; + $config{'Qstart'}=$config[7]; + $config{'Qlength'}=$config[8]; + $config{'Qoff'}=$config[9]; + $config{'Qon'}=$config[10]; + $config{'PaperID'}=$config[11]; + $config{'PaperIDlength'}=$config[12]; + $config{'FirstName'}=$config[13]; + $config{'FirstNamelength'}=$config[14]; + $config{'LastName'}=$config[15]; + $config{'LastNamelength'}=$config[16]; + $config{'BubblesPerRow'}=$config[17]; + last; + } + return %config; +} + =pod =item username_to_idmap @@ -6021,7 +6160,7 @@ sub username_to_idmap { Process a requested correction to a scanline. Arguments: - $scantron_config - hash from &Apache::lonnet::get_scantron_config() + $scantron_config - hash from &get_scantron_config() $scan_data - hash of correction information (see &scantron_getfile()) $line - existing scanline @@ -6704,7 +6843,7 @@ sub scantron_filter { sub scantron_process_corrections { my ($r) = @_; - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my ($scanlines,$scan_data)=&scantron_getfile(); my $classlist=&Apache::loncoursedata::get_classlist(); my $which=$env{'form.scantron_line'}; @@ -6873,7 +7012,7 @@ sub check_for_error { sub scantron_warning_screen { my ($button_text)=@_; my $title=&Apache::lonnet::gettitle($env{'form.selectpage'}); - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my $CODElist; if ($scantron_config{'CODElocation'} && $scantron_config{'CODEstart'} && @@ -7032,7 +7171,7 @@ sub scantron_validate_file { #get the student pick code ready $r->print(&Apache::loncommon::studentbrowser_javascript()); my $nav_error; - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my $max_bubble=&scantron_get_maxbubble(\$nav_error,\%scantron_config); if ($nav_error) { $r->print(&navmap_errormsg()); @@ -7491,7 +7630,7 @@ sub scantron_validate_ID { my %idmap=&username_to_idmap($classlist); #get scantron line setup - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my ($scanlines,$scan_data)=&scantron_getfile(); my $nav_error; @@ -7955,7 +8094,7 @@ sub prompt_for_corrections { Arguments: $r - Apache request object - $scan_config - hash from &Apache::lonnet::get_scantron_config() + $scan_config - hash from &get_scantron_config() $line - Number of the line being displayed. $questionnum - Question number (may include subquestion) $error - Type of error. @@ -8119,7 +8258,7 @@ sub get_codes { sub scantron_validate_CODE { my ($r,$currentphase) = @_; - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); if ($scantron_config{'CODElocation'} && $scantron_config{'CODEstart'} && $scantron_config{'CODElength'}) { @@ -8193,7 +8332,7 @@ sub scantron_validate_doublebubble { &Apache::lonnet::decode_symb($env{'form.selectpage'}); #get scantron line setup - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my ($scanlines,$scan_data)=&scantron_getfile(); my $navmap = Apache::lonnavmaps::navmap->new(); @@ -8375,7 +8514,7 @@ sub scantron_validate_missingbubbles { &Apache::lonnet::decode_symb($env{'form.selectpage'}); #get scantron line setup - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my ($scanlines,$scan_data)=&scantron_getfile(); my $navmap = Apache::lonnavmaps::navmap->new(); @@ -8504,7 +8643,7 @@ sub hand_bubble_option { } } if ($needs_hand_bubbles) { - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my $bubbles_per_row = &bubblesheet_bubbles_per_row(\%scantron_config); return &mt('The sequence to be graded contains response types which are handgraded.').'

'. &mt('If you have already graded these by bubbling sheets to indicate points awarded, [_1]what point value is assigned to a filled last bubble in each row?','
'). @@ -8524,7 +8663,7 @@ sub scantron_process_students { } my $default_form_data=&defaultFormData($symb); - my %scantron_config=&Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + my %scantron_config=&get_scantron_config($env{'form.scantron_format'}); my $bubbles_per_row = &bubblesheet_bubbles_per_row(\%scantron_config); my ($scanlines,$scan_data)=&scantron_getfile(); my $classlist=&Apache::loncoursedata::get_classlist(); @@ -8594,7 +8733,7 @@ SCANTRONFORM return ''; # Dunno why the other returns return '' rather than just returning. } - my %lettdig = &Apache::lonnet::letter_to_digits(); + my %lettdig = &letter_to_digits(); my $numletts = scalar(keys(%lettdig)); my %orderedforcode; @@ -8925,7 +9064,6 @@ sub grade_student_bubbles { sub scantron_upload_scantron_data { my ($r)=@_; my $dom = $env{'request.role.domain'}; - my ($formatoptions,$formattitle,$formatjs) = &scantron_upload_dataformat($dom); my $domdesc = &Apache::lonnet::domain($dom,'description'); $r->print(&Apache::loncommon::coursebrowser_javascript($dom)); my $select_link=&Apache::loncommon::selectcourse_link('rules','courseid', @@ -8939,7 +9077,8 @@ sub scantron_upload_scantron_data { &js_escape(\$nofile_alert); my $nocourseid_alert = &mt("Please use the 'Select Course' link to open a separate window where you can search for a course to which a file can be uploaded."); &js_escape(\$nocourseid_alert); - $r->print(&Apache::lonhtmlcommon::scripttag(' + $r->print(' + +

'.&mt('Send bubblesheet data to a course').'

@@ -8982,12 +9120,7 @@ sub scantron_upload_scantron_data { &Apache::lonhtmlcommon::row_closure(). &Apache::lonhtmlcommon::row_title(&mt('Domain')). ''.$domdesc. - &Apache::lonhtmlcommon::row_closure()); - if ($formatoptions) { - $r->print(&Apache::lonhtmlcommon::row_title($formattitle).$formatoptions. - &Apache::lonhtmlcommon::row_closure()); - } - $r->print( + &Apache::lonhtmlcommon::row_closure(). &Apache::lonhtmlcommon::row_title(&mt('File to upload')). ''. &Apache::lonhtmlcommon::row_closure(1). @@ -9000,84 +9133,6 @@ sub scantron_upload_scantron_data { return ''; } -sub scantron_upload_dataformat { - my ($dom) = @_; - my ($formatoptions,$formattitle,$formatjs); - $formatjs = <<'END'; -function toggleScantab(form) { - return; -} -END - my %domconfig = &Apache::lonnet::get_dom('configuration',['scantron'],$dom); - if (ref($domconfig{'scantron'}) eq 'HASH') { - if (ref($domconfig{'scantron'}{'config'}) eq 'HASH') { - if (keys(%{$domconfig{'scantron'}{'config'}}) > 1) { - if (($domconfig{'scantron'}{'config'}{'dat'}) && - (ref($domconfig{'scantron'}{'config'}{'csv'}) eq 'HASH')) { - if (ref($domconfig{'scantron'}{'config'}{'csv'}{'fields'}) eq 'HASH') { - if (keys(%{$domconfig{'scantron'}{'config'}{'csv'}{'fields'}})) { - my ($onclick,$formatextra,$singleline); - my @lines = &Apache::lonnet::get_scantronformat_file(); - my $count = 0; - foreach my $line (@lines) { - next if ($line =~ /^#/); - $singleline = $line; - $count ++; - } - if ($count > 1) { - $formatextra = ''; - $onclick = ' onclick="toggleScantab(this.form);"'; - $formatjs = <<"END"; -function toggleScantab(form) { - var divid = 'bubbletype'; - if (document.getElementById(divid)) { - var radioname = 'fileformat'; - var num = form.elements[radioname].length; - if (num) { - for (var i=0; i'; - } - $formattitle = &mt('File format'); - $formatoptions = ''.(' 'x2). - ''.$formatextra; - } - } - } - } elsif (keys(%{$domconfig{'scantron'}{'config'}}) == 1) { - if (ref($domconfig{'scantron'}{'config'}{'csv'}{'fields'}) eq 'HASH') { - if (keys(%{$domconfig{'scantron'}{'config'}{'csv'}{'fields'}})) { - $formattitle = &mt('Bubblesheet type'); - $formatoptions = &scantron_scantab(); - } - } - } - } - } - return ($formatoptions,$formattitle,$formatjs); -} sub scantron_upload_scantron_data_save { my($r)=@_; @@ -9107,38 +9162,8 @@ sub scantron_upload_scantron_data_save { &mt('The file: [_1] you attempted to upload contained no information. Please check that you entered the correct filename.', ''.&HTML::Entities::encode($env{'form.upfile.filename'},'<>&"').''),1)); } else { - my %domconfig = &Apache::lonnet::get_dom('configuration',['scantron'],$env{'form.domainid'}); - my $parser; - if (ref($domconfig{'scantron'}) eq 'HASH') { - if (ref($domconfig{'scantron'}{'config'}) eq 'HASH') { - my $is_csv; - my @possibles = keys(%{$domconfig{'scantron'}{'config'}}); - if (@possibles > 1) { - if ($env{'form.fileformat'} eq 'csv') { - if (ref($domconfig{'scantron'}{'config'}{'csv'}) eq 'HASH') { - if (ref($domconfig{'scantron'}{'config'}{'csv'}{'fields'}) eq 'HASH') { - if (keys(%{$domconfig{'scantron'}{'config'}{'csv'}{'fields'}}) > 1) { - $is_csv = 1; - } - } - } - } - } elsif (@possibles == 1) { - if (ref($domconfig{'scantron'}{'config'}{'csv'}) eq 'HASH') { - if (ref($domconfig{'scantron'}{'config'}{'csv'}{'fields'}) eq 'HASH') { - if (keys(%{$domconfig{'scantron'}{'config'}{'csv'}{'fields'}}) > 1) { - $is_csv = 1; - } - } - } - } - if ($is_csv) { - $parser = $domconfig{'scantron'}{'config'}{'csv'}; - } - } - } - my $result = - &Apache::lonnet::userfileupload('upfile','scantron','scantron',$parser,'','', + my $result = + &Apache::lonnet::userfileupload('upfile','','scantron','','','', $env{'form.courseid'},$env{'form.domainid'}); if ($result =~ m{^/uploaded/}) { $r->print( @@ -9183,7 +9208,7 @@ sub validate_uploaded_scantron_file { $idmap{$lckey} = $idmap{$key}; } my %unique_formats; - my @formatlines = &Apache::lonnet::get_scantronformat_file(); + my @formatlines = &get_scantronformat_file(); foreach my $line (@formatlines) { chomp($line); my @config = split(/:/,$line); @@ -9329,14 +9354,14 @@ sub checkscantron_results { if (!$symb) {return '';} my $grading_menu_button=&show_grading_menu_form($symb); my $cid = $env{'request.course.id'}; - my %lettdig = &Apache::lonnet::letter_to_digits(); + my %lettdig = &letter_to_digits(); my $numletts = scalar(keys(%lettdig)); my $cnum = $env{'course.'.$cid.'.num'}; my $cdom = $env{'course.'.$cid.'.domain'}; my (undef, undef, $sequence) = &Apache::lonnet::decode_symb($env{'form.selectpage'}); my %record; my %scantron_config = - &Apache::lonnet::get_scantron_config($env{'form.scantron_format'}); + &Apache::grades::get_scantron_config($env{'form.scantron_format'}); my $bubbles_per_row = &bubblesheet_bubbles_per_row(\%scantron_config); my ($scanlines,$scan_data)=&Apache::grades::scantron_getfile(); my $classlist=&Apache::loncoursedata::get_classlist(); @@ -9660,6 +9685,23 @@ sub verify_scantron_grading { return ($counter,$record); } +sub letter_to_digits { + my %lettdig = ( + A => 1, + B => 2, + C => 3, + D => 4, + E => 5, + F => 6, + G => 7, + H => 8, + I => 9, + J => 0, + ); + return %lettdig; +} + + #-------- end of section for handling grading scantron forms ------- # #------------------------------------------------------------------- @@ -10264,7 +10306,7 @@ sub process_clicker_file { $result .= &Apache::lonhtmlcommon::confirm_success( &mt('No IDs found to determine correct answer'),1); - return $result.&show_grading_menu_form($symb); + return $result,.&show_grading_menu_form($symb); } } if (length($env{'form.upfile'}) < 2) { @@ -10274,22 +10316,6 @@ sub process_clicker_file { ''.&HTML::Entities::encode($env{'form.upfile.filename'},'<>&"').''),1); return $result.&show_grading_menu_form($symb); } - my $mimetype; - if ($env{'form.upfiletype'} eq 'iclicker') { - my $mm = new File::MMagic; - $mimetype = $mm->checktype_contents($env{'form.upfile'}); - unless (($mimetype eq 'text/plain') || ($mimetype eq 'text/html')) { - $result.= '

'. - &Apache::lonhtmlcommon::confirm_success( - &mt('File format is neither csv (iclicker 6) nor xml (iclicker 7)'),1).'

'; - return $result.&show_grading_menu_form($symb); - } - } elsif (($env{'form.upfiletype'} ne 'interwrite') && ($env{'form.upfiletype'} ne 'turning')) { - $result .= '

'. - &Apache::lonhtmlcommon::confirm_success( - &mt('Invalid clicker type: choose one of: i>clicker, Interwrite PRS, or Turning Technologies.'),1).'

'; - return $result.&show_grading_menu_form($symb); - } # Were able to get all the info needed, now analyze the file @@ -10317,14 +10343,12 @@ ENDHEADER my $errormsg=''; my $number=0; if ($env{'form.upfiletype'} eq 'iclicker') { - if ($mimetype eq 'text/plain') { - ($errormsg,$number)=&iclicker_eval(\@questiontitles,\%responses); - } elsif ($mimetype eq 'text/html') { - ($errormsg,$number)=&iclickerxml_eval(\@questiontitles,\%responses); - } - } elsif ($env{'form.upfiletype'} eq 'interwrite') { + ($errormsg,$number)=&iclicker_eval(\@questiontitles,\%responses); + } + if ($env{'form.upfiletype'} eq 'interwrite') { ($errormsg,$number)=&interwrite_eval(\@questiontitles,\%responses); - } elsif ($env{'form.upfiletype'} eq 'turning') { + } + if ($env{'form.upfiletype'} eq 'turning') { ($errormsg,$number)=&turning_eval(\@questiontitles,\%responses); } $result.='
'.&mt('Found [_1] question(s)',$number).'
'. @@ -10427,49 +10451,6 @@ sub iclicker_eval { return ($errormsg,$number); } -sub iclickerxml_eval { - my ($questiontitles,$responses)=@_; - my $number=0; - my $errormsg=''; - my @state; - my %respbyid; - my $p = HTML::Parser->new - ( - xml_mode => 1, - start_h => - [sub { - my ($tagname,$attr) = @_; - push(@state,$tagname); - if ("@state" eq "ssn p") { - my $title = $attr->{qn}; - $title =~ s/(^\s+|\s+$)//g; - $questiontitles->[$number]=$title; - } elsif ("@state" eq "ssn p v") { - my $id = $attr->{id}; - my $entry = $attr->{ans}; - $id=~s/^[\#0]+//; - $entry =~s/[^a-zA-Z0-9\.\*\-\+]+//g; - $respbyid{$id}[$number] = $entry; - } - }, "tagname, attr"], - end_h => - [sub { - my ($tagname) = @_; - if ("@state" eq "ssn p") { - $number++; - } - pop(@state); - }, "tagname"], - ); - - $p->parse($env{'form.upfile'}); - $p->eof; - foreach my $id (keys(%respbyid)) { - $responses->{$id}=join(',',@{$respbyid{$id}}); - } - return ($errormsg,$number); -} - sub interwrite_eval { my ($questiontitles,$responses)=@_; my $number=0; @@ -10668,20 +10649,12 @@ sub navmap_errormsg { } sub startpage { - my ($r,$symb,$crumbs,$onlyfolderflag,$nodisplayflag,$stuvcurrent,$stuvdisp,$nomenu,$js,$onload) = @_; - my %args; - if ($onload) { - my %loaditems = ( - 'onload' => $onload, - ); - $args{'add_entries'} = \%loaditems; - } + my ($r,$symb,$crumbs,$onlyfolderflag,$nodisplayflag,$stuvcurrent,$stuvdisp,$nomenu,$js) = @_; if ($nomenu) { - $args{'only_body'} = 1; - $r->print(&Apache::loncommon::start_page("Student's Version",$js,\%args)); + $r->print(&Apache::loncommon::start_page("Student's Version",$js,{'only_body' => '1'})); } else { - $args{'bread_crumbs'} = $crumbs; - $r->print(&Apache::loncommon::start_page('Grading',$js,\%args)); + $r->print(&Apache::loncommon::start_page('Grading',$js, + {'bread_crumbs' => $crumbs})); } unless ($nodisplayflag) { $r->print(&Apache::lonhtmlcommon::resource_info_box($symb,$onlyfolderflag,$stuvcurrent,$stuvdisp)); @@ -10763,16 +10736,8 @@ sub handler { } &Apache::loncommon::content_type($request,'text/html'); $request->send_http_header; - if (($command eq 'scantron_selectphase' && $perm{'mgr'}) || - (($command eq 'scantronupload') && - (&Apache::lonnet::allowed('usc',$env{'request.role.domain'}) || - &Apache::lonnet::allowed('usc',$env{'request.course.id'})))) { - &startpage($request,$symb,[{href=>'/adm/grades', text=>"Grading"}],1,1, - undef,undef,undef,undef,'toggleScantab(document.rules);'); - } else { - unless ((($command eq 'submission' || $command eq 'versionsub')) && ($perm{'vgr'})) { - $request->print($start_page); - } + unless ((($command eq 'submission' || $command eq 'versionsub')) && ($perm{'vgr'})) { + $request->print($start_page); } if ($command eq 'submission' && $perm{'vgr'}) { my ($stuvcurrent,$stuvdisp,$versionform,$js); @@ -10951,7 +10916,7 @@ ssi_with_retries() $r - Apache request object $i - number of the current scanline $scan_record - hash ref as returned from &scantron_parse_scanline() - $scan_config - hash ref as returned from &Apache::lonnet::get_scantron_config() + $scan_config - hash ref as returned from &get_scantron_config() $line - full contents of the current scanline $error - error condition, valid values are 'incorrectCODE', 'duplicateCODE',