Annotation of loncom/interface/slotrequest.pm, revision 1.23
1.1 albertel 1: # The LearningOnline Network with CAPA
2: # Handler for requesting to have slots added to a students record
3: #
1.23 ! albertel 4: # $Id: slotrequest.pm,v 1.22 2005/10/14 20:00:34 albertel Exp $
1.1 albertel 5: #
6: # Copyright Michigan State University Board of Trustees
7: #
8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
9: #
10: # LON-CAPA is free software; you can redistribute it and/or modify
11: # it under the terms of the GNU General Public License as published by
12: # the Free Software Foundation; either version 2 of the License, or
13: # (at your option) any later version.
14: #
15: # LON-CAPA is distributed in the hope that it will be useful,
16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18: # GNU General Public License for more details.
19: #
20: # You should have received a copy of the GNU General Public License
21: # along with LON-CAPA; if not, write to the Free Software
22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23: #
24: # /home/httpd/html/adm/gpl.txt
25: #
26: # http://www.lon-capa.org/
27: #
28: ###
29:
30: package Apache::slotrequest;
31:
32: use strict;
33: use Apache::Constants qw(:common :http :methods);
34: use Apache::loncommon();
35: use Apache::lonlocal;
36: use Apache::lonnet;
1.14 albertel 37: use Date::Manip;
1.1 albertel 38:
39: sub fail {
40: my ($r,$code)=@_;
1.2 albertel 41: if ($code eq 'not_valid') {
1.8 albertel 42: $r->print('<p>'.&mt('Unable to understand what resource you wanted to sign up for.').'</p>');
1.2 albertel 43:
1.8 albertel 44: } elsif ($code eq 'not_allowed') {
45: $r->print('<p>'.&mt('Not allowed to sign up or change reservations at this time.').'</p>');
46: } else {
47: $r->print('<p>'.&mt('Failed.').'</p>');
1.2 albertel 48: }
1.8 albertel 49:
1.2 albertel 50: $r->print('<p><a href="/adm/flip?postdata=return:">'.
51: &mt('Return to last resource').'</a></p>');
1.1 albertel 52: &end_page($r);
53: }
54:
55: sub start_page {
56: my ($r)=@_;
57: my $html=&Apache::lonxml::xmlbegin();
58: $r->print($html.'<head><title>'.
59: &mt('Request another Worktime').'</title></head>');
60: $r->print(&Apache::loncommon::bodytag('Requesting another Worktime'));
61: }
62:
63: sub end_page {
64: my ($r)=@_;
65: $r->print(&Apache::loncommon::endbodytag().'</html>');
66: }
67:
1.2 albertel 68: =pod
69:
70: slot_reservations db
71: - keys are
72: - slotname\0id -> value is an hashref of
73: name -> user@domain of holder
74: timestamp -> timestamp of reservation
75: symb -> symb of resource that it is reserved for
76:
77: =cut
78:
79: sub get_course {
80: (undef,my $courseid)=&Apache::lonxml::whichuser();
81: my $cdom=$env{'course.'.$courseid.'.domain'};
82: my $cnum=$env{'course.'.$courseid.'.num'};
83: return ($cnum,$cdom);
84: }
85:
86: sub get_reservation_ids {
87: my ($slot_name)=@_;
88:
89: my ($cnum,$cdom)=&get_course();
90:
91: my %consumed=&Apache::lonnet::dump('slot_reservations',$cdom,$cnum,
92: "^$slot_name\0");
93:
94: my ($tmp)=%consumed;
95: if ($tmp=~/^error: 2 / ) {
96: return 0;
97: }
98: return keys(%consumed);
99: }
100:
101: sub space_available {
102: my ($slot_name,$slot)=@_;
103: my $max=$slot->{'maxspace'};
104:
105: if (!defined($max)) { return 1; }
106:
107: my $consumed=scalar(&get_reservation_ids($slot_name));
108: if ($consumed < $max) {
109: return 1
110: }
111: return 0;
112: }
1.3 albertel 113:
1.4 albertel 114: sub check_for_reservation {
115: my ($symb)=@_;
116: my $student = &Apache::lonnet::EXT("resource.0.availablestudent", $symb,
117: $env{'user.domain'}, $env{'user.name'});
118:
119: my $course = &Apache::lonnet::EXT("resource.0.available", $symb,
120: $env{'user.domain'}, $env{'user.name'});
121: my @slots = (split(/:/,$student), split(/:/, $course));
122:
123: &Apache::lonxml::debug(" slot list is ".join(':',@slots));
124:
125: my ($cnum,$cdom)=&get_course();
126: my %slots=&Apache::lonnet::get('slots', [@slots], $cdom, $cnum);
127:
128: foreach my $slot_name (@slots) {
129: next if (!defined($slots{$slot_name}) ||
130: !ref($slots{$slot_name}));
131: &Apache::lonxml::debug(time." $slot_name ".
132: $slots{$slot_name}->{'starttime'}." -- ".
133: $slots{$slot_name}->{'startreserve'});
1.7 albertel 134: if ($slots{$slot_name}->{'endtime'} > time &&
1.4 albertel 135: $slots{$slot_name}->{'startreserve'} < time) {
1.7 albertel 136: # between start of reservation times and end of slot
1.4 albertel 137: return($slot_name, $slots{$slot_name});
138: }
139: }
140: return (undef,undef);
141: }
142:
1.5 albertel 143: sub check_for_conflict {
144: my ($symb,$new_slot_name)=@_;
145: my $student = &Apache::lonnet::EXT("resource.0.availablestudent", $symb,
146: $env{'user.domain'}, $env{'user.name'});
147: my $course = &Apache::lonnet::EXT("resource.0.available", $symb,
148: $env{'user.domain'}, $env{'user.name'});
149: my @slots = (split(/:/,$student), split(/:/, $course));
150: my ($cnum,$cdom)=&get_course();
151: my %slots=&Apache::lonnet::get('slots', [@slots], $cdom, $cnum);
152: foreach my $slot_name (@slots) {
153: next if (!defined($slots{$slot_name}) ||
154: !ref($slots{$slot_name}));
155:
156: next if (!defined($slots{$slot_name}->{'uniqueperiod'}) ||
157: !ref($slots{$slot_name}->{'uniqueperiod'}));
158: my ($start,$end)=@{$slots{$slot_name}->{'uniqueperiod'}};
159: if ($start<time && time < $end) {
160: return $slot_name;
161: }
162: }
163: return undef;
164:
165: }
166:
1.2 albertel 167: sub make_reservation {
168: my ($slot_name,$slot,$symb)=@_;
1.3 albertel 169:
170: my ($cnum,$cdom)=&get_course();
171:
172: my $value=&Apache::lonnet::EXT("resource.0.availablestudent",$symb,
173: $env{'user.domain'},$env{'user.name'});
174: &Apache::lonxml::debug("value is $value<br />");
175: foreach my $other_slot (split(/:/, $value)) {
176: if ($other_slot eq $slot_name) {
177: my %consumed=&Apache::lonnet::dump('slot_reservations', $cdom,
178: $cnum, "^$slot_name\0");
179:
180: my $me=$env{'user.name'}.'@'.$env{'user.domain'};
181: foreach my $key (keys(%consumed)) {
182: if ($consumed{$key}->{'name'} eq $me) {
183: my $num=(split('\0',$key))[1];
184: return -$num;
185: }
186: }
187: }
188: }
189:
1.2 albertel 190: my $max=$slot->{'maxspace'};
1.3 albertel 191: if (!defined($max)) { $max=99999; }
1.2 albertel 192:
193: my (@ids)=&get_reservation_ids($slot_name);
194:
195: my $last=0;
196: foreach my $id (@ids) {
197: my $num=(split('\0',$id))[1];
198: if ($num > $last) { $last=$num; }
199: }
200:
201: my $wanted=$last+1;
1.3 albertel 202: &Apache::lonxml::debug("wanted $wanted<br />");
1.7 albertel 203: if (scalar(@ids) >= $max) {
1.2 albertel 204: # full up
1.7 albertel 205: return undef;
1.2 albertel 206: }
207:
208: my %reservation=('name' => $env{'user.name'}.'@'.$env{'user.domain'},
209: 'timestamp' => time,
210: 'symb' => $symb);
211:
212: my $success=&Apache::lonnet::newput('slot_reservations',
213: {"$slot_name\0$wanted" =>
214: \%reservation},
1.3 albertel 215: $cdom, $cnum);
216:
1.2 albertel 217: if ($success eq 'ok') {
1.3 albertel 218: my $new_value=$slot_name;
219: if ($value) {
220: $new_value=$value.':'.$new_value;
221: }
222: my $result=&Apache::lonparmset::storeparm_by_symb($symb,
223: '0_availablestudent',
224: 1, $new_value, 'string',
225: $env{'user.name'},
226: $env{'user.domain'});
227: &Apache::lonxml::debug("hrrm $result");
1.2 albertel 228: return $wanted;
229: }
1.3 albertel 230:
1.2 albertel 231: # someone else got it
1.3 albertel 232: return undef;
233: }
234:
1.5 albertel 235: sub release_slot {
1.6 albertel 236: my ($r,$symb,$slot_name,$inhibit_return_link)=@_;
237:
238: if ($slot_name eq '') { $slot_name=$env{'form.slotname'}; }
239: my ($cnum,$cdom)=&get_course();
240:
1.5 albertel 241: # get parameter string, check for existance, rebuild string with the slot
1.6 albertel 242:
243: my @slots = split(/:/,&Apache::lonnet::EXT("resource.0.availablestudent",
244: $symb,$env{'user.domain'},
245: $env{'user.name'}));
246: my @new_slots;
247: foreach my $exist_slot (@slots) {
248: if ($exist_slot eq $slot_name) { next; }
249: push(@new_slots,$exist_slot);
250: }
251: my $new_param = join(':',@new_slots);
1.5 albertel 252:
253: # get slot reservations, check if user has one, if so remove reservation
1.6 albertel 254: my %consumed=&Apache::lonnet::dump('slot_reservations',$cdom,$cnum,
255: "^$slot_name\0");
256: foreach my $entry (keys(%consumed)) {
257: if ( $consumed{$entry}->{'name'} eq
258: ($env{'user.name'}.'@'.$env{'user.domain'}) ) {
259: &Apache::lonnet::del('slot_reservations',[$entry],
260: $cdom,$cnum);
261: }
262: }
1.5 albertel 263: # store new parameter string
1.6 albertel 264: my $result=&Apache::lonparmset::storeparm_by_symb($symb,
265: '0_availablestudent',
266: 1, $new_param, 'string',
267: $env{'user.name'},
268: $env{'user.domain'});
269: my %slot=&Apache::lonnet::get_slot($slot_name);
270: my $description=&get_description($env{'form.slotname'},\%slot);
271: $r->print("<p>Released Reservation: $description</p>");
1.7 albertel 272: if (!$inhibit_return_link) {
1.6 albertel 273: $r->print('<p><a href="/adm/flip?postdata=return:">'.
274: &mt('Return to last resource').'</a></p>');
275: }
276: return 1;
1.5 albertel 277: }
278:
1.3 albertel 279: sub get_slot {
280: my ($r,$symb)=@_;
281:
1.5 albertel 282: my $slot_name=&check_for_conflict($symb,$env{'form.slotname'});
283: if ($slot_name) {
284: my %slot=&Apache::lonnet::get_slot($slot_name);
1.6 albertel 285: my $description1=&get_description($slot_name,\%slot);
286: %slot=&Apache::lonnet::get_slot($env{'form.slotname'});
287: my $description2=&get_description($env{'form.slotname'},\%slot);
288: $r->print("<p>Already have a reservation: $description1</p>");
1.7 albertel 289: if ($slot_name ne $env{'form.slotname'}) {
290: $r->print(<<STUFF);
1.6 albertel 291: <form method="POST" action="/adm/slotrequest">
292: <input type="hidden" name="symb" value="$env{'form.symb'}" />
293: <input type="hidden" name="slotname" value="$env{'form.slotname'}" />
294: <input type="hidden" name="releaseslot" value="$slot_name" />
295: <input type="hidden" name="command" value="change" />
296: STUFF
1.7 albertel 297: $r->print("<p>You can either ");
298: $r->print(<<STUFF);
1.6 albertel 299: <input type="submit" name="change" value="Change" />
300: STUFF
1.7 albertel 301: $r->print(' your reservation from <b>'.$description1.'</b> to <b>'.
302: $description2.
303: '</b> <br />or <a href="/adm/flip?postdata=return:">'.
304: &mt('Return to last resource').'</a></p>');
305: $r->print(<<STUFF);
1.6 albertel 306: </form>
307: STUFF
1.7 albertel 308: } else {
309: $r->print('<p><a href="/adm/flip?postdata=return:">'.
310: &mt('Return to last resource').'</a></p>');
311: }
1.5 albertel 312: return;
313: }
1.3 albertel 314: my %slot=&Apache::lonnet::get_slot($env{'form.slotname'});
315: my $reserved=&make_reservation($env{'form.slotname'},
316: \%slot,$symb);
317: my $description=&get_description($env{'form.slotname'},\%slot);
1.7 albertel 318: if (defined($reserved)) {
319: if ($reserved > -1) {
320: $r->print("<p>Success: $description</p>");
321: $r->print('<p><a href="/adm/flip?postdata=return:">'.
322: &mt('Return to last resource').'</a></p>');
323: return;
324: } elsif ($reserved < 0) {
325: $r->print("<p>Already reserved: $description</p>");
326: $r->print('<p><a href="/adm/flip?postdata=return:">'.
327: &mt('Return to last resource').'</a></p>');
328: return;
329: }
1.3 albertel 330: }
331:
1.7 albertel 332: my %lt=('request'=>"Availibility list",
1.3 albertel 333: 'try' =>'Try again');
334: %lt=&Apache::lonlocal::texthash(%lt);
335:
336: $r->print(<<STUFF);
337: <p> <font color="red">Failed</font> to reserve a spot for $description. </p>
338: <p>
339: <form method="POST" action="/adm/slotrequest">
340: <input type="submit" name="Try Again" value="$lt{'try'}" />
341: <input type="hidden" name="symb" value="$env{'form.symb'}" />
342: <input type="hidden" name="slotname" value="$env{'form.slotname'}" />
343: <input type="hidden" name="command" value="get" />
344: </form>
345: ?
346: </p>
347: <p>
348: or
349: <form method="POST" action="/adm/slotrequest">
350: <input type="hidden" name="symb" value="$env{'form.symb'}" />
351: <input type="submit" name="requestattempt" value="$lt{'request'}" />
352: </form>
353: </p>
354: or
355: STUFF
356: $r->print('<p><a href="/adm/flip?postdata=return:">'.
357: &mt('Return to last resource').'</a></p>');
358: return;
359: }
360:
361: sub allowed_slot {
362: my ($slot_name,$slot,$symb)=@_;
363: #already started
364: if ($slot->{'starttime'} < time) {
1.5 albertel 365: # all open slot to be schedulable
366: #return 0;
1.3 albertel 367: }
1.5 albertel 368: &Apache::lonxml::debug("$slot_name starttime good");
1.3 albertel 369: #already ended
370: if ($slot->{'endtime'} < time) {
371: return 0;
372: }
1.5 albertel 373: &Apache::lonxml::debug("$slot_name endtime good");
1.3 albertel 374: # not allowed to pick this one
375: if (defined($slot->{'type'})
376: && $slot->{'type'} ne 'schedulable_student') {
377: return 0;
378: }
1.5 albertel 379: &Apache::lonxml::debug("$slot_name type good");
1.3 albertel 380: # not allowed for this resource
381: if (defined($slot->{'symb'})
382: && $slot->{'symb'} ne $symb) {
383: return 0;
384: }
1.5 albertel 385: &Apache::lonxml::debug("$slot_name symb good");
1.3 albertel 386: return 1;
1.2 albertel 387: }
388:
1.3 albertel 389: sub get_description {
390: my ($slot_name,$slot)=@_;
391: my $description=$slot->{'description'};
392: if (!defined($description)) {
1.4 albertel 393: $description=&mt('[_1] From [_2] to [_3]',$slot_name,
1.3 albertel 394: &Apache::lonlocal::locallocaltime($slot->{'starttime'}),
395: &Apache::lonlocal::locallocaltime($slot->{'endtime'}));
396: }
397: return $description;
398: }
1.2 albertel 399:
400: sub show_choices {
401: my ($r,$symb)=@_;
402:
403: my ($cnum,$cdom)=&get_course();
404: my %slots=&Apache::lonnet::dump('slots',$cdom,$cnum);
1.3 albertel 405: my $available;
1.2 albertel 406: $r->print('<table border="1">');
1.5 albertel 407: &Apache::lonxml::debug("Checking Slots");
408: my ($got_slot)=&check_for_reservation($symb);
1.2 albertel 409: foreach my $slot (sort
410: { return $slots{$a}->{'starttime'} <=> $slots{$b}->{'starttime'} }
411: (keys(%slots))) {
1.5 albertel 412:
413: &Apache::lonxml::debug("Checking Slot $slot");
1.3 albertel 414: next if (!&allowed_slot($slot,$slots{$slot}));
415:
416: $available++;
417:
418: my $description=&get_description($slot,$slots{$slot});
1.2 albertel 419:
420: my $form=&mt('Unavailable');
1.7 albertel 421: if (($slot eq $got_slot) ||
422: &space_available($slot,$slots{$slot},$symb)) {
1.5 albertel 423: my $text=&mt('Select');
424: my $command='get';
425: if ($slot eq $got_slot) {
426: $text=&mt('Free Reservation');
427: $command='release';
428: }
1.3 albertel 429: my $escsymb=&Apache::lonnet::escape($symb);
1.2 albertel 430: $form=<<STUFF;
1.3 albertel 431: <form method="POST" action="/adm/slotrequest">
1.5 albertel 432: <input type="submit" name="Select" value="$text" />
1.3 albertel 433: <input type="hidden" name="symb" value="$escsymb" />
434: <input type="hidden" name="slotname" value="$slot" />
1.5 albertel 435: <input type="hidden" name="command" value="$command" />
1.2 albertel 436: </form>
437: STUFF
438: }
439: $r->print(<<STUFF);
440: <tr>
441: <td>$form</td>
442: <td>$description</td>
443: </tr>
444: STUFF
445: }
1.3 albertel 446:
447: if (!$available) {
1.5 albertel 448: $r->print('<tr><td>No available times. <a href="/adm/flip?postdata=return:">'.
1.3 albertel 449: &mt('Return to last resource').'</a></td></tr>');
450: }
1.2 albertel 451: $r->print('</table>');
452: }
453:
1.5 albertel 454: sub show_table {
1.19 albertel 455: my ($r,$mgr)=@_;
1.5 albertel 456:
457: my ($cnum,$cdom)=&get_course();
458: my %slots=&Apache::lonnet::dump('slots',$cdom,$cnum);
1.19 albertel 459: if ( (keys(%slots))[0] =~ /^error: 2 /) {
460: undef(%slots);
461: }
1.5 albertel 462: my $available;
1.14 albertel 463: if ($mgr eq 'F') {
464: $r->print('<form method="POST" action="/adm/slotrequest">
465: <input type="hidden" name="command" value="uploadstart" />
466: <input type="submit" name="start" value="'.&mt('Upload Slot List').'" />
467: </form>');
468: }
1.21 albertel 469: my $linkstart='<a href="/adm/slotrequest?command=showslots&order=';
1.10 albertel 470: $r->print('<table border="1">
471: <tr>
1.19 albertel 472: <th></th>
1.23 ! albertel 473: <th>'.$linkstart.'name" >Slot name</a></th>
! 474: <th>'.$linkstart.'type" >Type</a></th>
! 475: <th>'.$linkstart.'description" >Description</a></th>
! 476: <th>'.$linkstart.'starttime" >Start Time</a></th>
! 477: <th>'.$linkstart.'endtime" >End Time</a></th>
! 478: <th>'.$linkstart.'startreserve">Time Students Can Start Reserving</a></th>
! 479: <th>'.$linkstart.'secret" >Secret</a></th>
! 480: <th>'.$linkstart.'maxspace" >Max space</a></th>
! 481: <th> Scheduled Students</th>
! 482: <th>'.$linkstart.'unique" >Unique Period</a></th>
1.10 albertel 483: </tr>');
1.21 albertel 484: my %name_cache;
485: my $slotsort = sub {
1.23 ! albertel 486: if ($env{'form.order'}=~/^(type|description|endtime|maxspace)$/) {
1.21 albertel 487: if (lc($slots{$a}->{$env{'form.order'}})
488: ne lc($slots{$b}->{$env{'form.order'}})) {
489: return (lc($slots{$a}->{$env{'form.order'}})
490: cmp lc($slots{$b}->{$env{'form.order'}}));
491: }
1.23 ! albertel 492: } elsif ($env{'form.order'} eq 'name') {
! 493: if (lc($a) cmp lc($b)) {
! 494: return lc($a) cmp lc($b);
! 495: }
1.21 albertel 496: } elsif ($env{'form.order'} eq 'unique') {
497:
498: if ($slots{$a}->{'uniqueperiod'}[0]
499: ne $slots{$b}->{'uniqueperiod'}[0]) {
500: return ($slots{$a}->{'uniqueperiod'}[0]
501: cmp $slots{$b}->{'uniqueperiod'}[0]);
502: }
503: if ($slots{$a}->{'uniqueperiod'}[1]
504: ne $slots{$b}->{'uniqueperiod'}[1]) {
505: return ($slots{$a}->{'uniqueperiod'}[1]
506: cmp $slots{$b}->{'uniqueperiod'}[1]);
507: }
508: }
509: return $slots{$a}->{'starttime'} <=> $slots{$b}->{'starttime'};
510: };
511: foreach my $slot (sort $slotsort (keys(%slots))) {
1.5 albertel 512: if (defined($slots{$slot}->{'type'})
513: && $slots{$slot}->{'type'} ne 'schedulable_student') {
1.13 albertel 514: #next;
1.5 albertel 515: }
516: my $description=&get_description($slot,$slots{$slot});
517: my %consumed=&Apache::lonnet::dump('slot_reservations',$cdom,$cnum,
518: "^$slot\0");
519: my $ids;
520: foreach my $entry (sort(keys(%consumed))) {
521: my (undef,$id)=split("\0",$entry);
522: $ids.= $id.'-> '.$consumed{$entry}->{'name'}.'<br />';
523: }
1.23 ! albertel 524: my $start=&Apache::lonlocal::locallocaltime($slots{$slot}->{'starttime'});
! 525: my $end=&Apache::lonlocal::locallocaltime($slots{$slot}->{'endtime'});
! 526: my $start_reserve=&Apache::lonlocal::locallocaltime($slots{$slot}->{'startreserve'});
1.14 albertel 527: my $unique;
528: if (ref($slots{$slot}{'uniqueperiod'})) {
529: $unique=localtime($slots{$slot}{'uniqueperiod'}[0]).','.
530: localtime($slots{$slot}{'uniqueperiod'}[1]);
531: }
1.20 albertel 532: my @proctors = map {
533: my ($uname,$udom)=split(/@/,$_);
1.21 albertel 534: my $fullname=$name_cache{$_};
535: if (!defined($fullname)) {
536: &Apache::lonnet::logthis("Gettign $uname $udom");
537: $fullname = &Apache::loncommon::plainname($uname,$udom);
538: $fullname =~s/\s/ /g;
539: $name_cache{$_} = $fullname;
540: }
1.20 albertel 541: &Apache::loncommon::aboutmewrapper($fullname,$uname,$udom);
542: } (split(/\s*,\s*/,$slots{$slot}->{'proctor'}));
543:
544: my $proctors=join(', ',@proctors);
1.14 albertel 545:
1.19 albertel 546: my $edit=(<<EDITFORM);
547: <form method="POST">
548: <input type="hidden" name="command" value="editslot" />
549: <input type="hidden" name="slot" value="$slot" />
550: <input type="submit" name="Edit" value="Edit" />
551: </form>
552: EDITFORM
1.5 albertel 553: $r->print(<<STUFF);
554: <tr>
1.21 albertel 555: <td rowspan="2">$edit</td>
1.5 albertel 556: <td>$slot</td>
1.10 albertel 557: <td>$slots{$slot}->{'type'}</td>
1.5 albertel 558: <td>$description</td>
559: <td>$start</td>
560: <td>$end</td>
1.23 ! albertel 561: <td>$start_reserve</td>
1.22 albertel 562: <td>$slots{$slot}->{'secret'}</td>
1.5 albertel 563: <td>$slots{$slot}->{'maxspace'}</td>
564: <td>$ids</td>
1.14 albertel 565: <td>$unique</td>
1.5 albertel 566: </tr>
1.21 albertel 567: <tr>
1.23 ! albertel 568: <td colspan="10">$proctors</td>
1.21 albertel 569: </tr>
1.5 albertel 570: STUFF
571: }
572: $r->print('</table>');
573: }
574:
1.14 albertel 575: sub upload_start {
1.19 albertel 576: my ($r)=@_;
1.14 albertel 577: $r->print(&Apache::grades::checkforfile_js());
578: my $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";
579: $result.=' <b>'.
580: &mt('Specify a file containing the slot definitions.').
581: '</b></td></tr>'."\n";
582: $result.='<tr bgcolor=#ffffe6><td>'."\n";
583: my $upfile_select=&Apache::loncommon::upfile_select_html();
584: my $ignore=&mt('Ignore First Line');
585: $result.=<<ENDUPFORM;
586: <form method="post" enctype="multipart/form-data" action="/adm/slotrequest" name="slotupload">
587: <input type="hidden" name="command" value="csvuploadmap" />
588: $upfile_select
589: <br /><input type="button" onClick="javascript:checkUpload(this.form);" value="Upload Data" />
590: <label><input type="checkbox" name="noFirstLine" />$ignore</label>
591: </form>
592: ENDUPFORM
593: $result.='</td></tr></table>'."\n";
594: $result.='</td></tr></table>'."\n";
595: $r->print($result);
596: }
597:
598: sub csvuploadmap_header {
1.19 albertel 599: my ($r,$datatoken,$distotal)= @_;
1.14 albertel 600: my $javascript;
601: if ($env{'form.upfile_associate'} eq 'reverse') {
602: $javascript=&csvupload_javascript_reverse_associate();
603: } else {
604: $javascript=&csvupload_javascript_forward_associate();
605: }
606:
607: my $checked=(($env{'form.noFirstLine'})?' checked="checked"':'');
608: my $ignore=&mt('Ignore First Line');
609: $r->print(<<ENDPICK);
610: <form method="post" enctype="multipart/form-data" action="/adm/slotrequest" name="slotupload">
611: <h3>Identify fields</h3>
612: Total number of records found in file: $distotal <hr />
613: Enter as many fields as you can. The system will inform you and bring you back
614: to this page if the data selected is insufficient to create the slots.<hr />
615: <input type="button" value="Reverse Association" onClick="javascript:this.form.associate.value='Reverse Association';submit(this.form);" />
616: <label><input type="checkbox" name="noFirstLine" $checked />$ignore</label>
617: <input type="hidden" name="associate" value="" />
618: <input type="hidden" name="datatoken" value="$datatoken" />
619: <input type="hidden" name="fileupload" value="$env{'form.fileupload'}" />
620: <input type="hidden" name="upfiletype" value="$env{'form.upfiletype'}" />
621: <input type="hidden" name="upfile_associate"
622: value="$env{'form.upfile_associate'}" />
623: <input type="hidden" name="command" value="csvuploadassign" />
624: <hr />
625: <script type="text/javascript" language="Javascript">
626: $javascript
627: </script>
628: ENDPICK
629: return '';
630:
631: }
632:
633: sub csvuploadmap_footer {
634: my ($request,$i,$keyfields) =@_;
635: $request->print(<<ENDPICK);
636: </table>
637: <input type="hidden" name="nfields" value="$i" />
638: <input type="hidden" name="keyfields" value="$keyfields" />
639: <input type="button" onClick="javascript:verify(this.form)" value="Create Slots" /><br />
640: </form>
641: ENDPICK
642: }
643:
644: sub csvupload_javascript_reverse_associate {
645: my $error1=&mt('You need to specify the name, starttime, endtime and a type');
646: return(<<ENDPICK);
647: function verify(vf) {
648: var foundstart=0;
649: var foundend=0;
650: var foundname=0;
651: var foundtype=0;
652: for (i=0;i<=vf.nfields.value;i++) {
653: tw=eval('vf.f'+i+'.selectedIndex');
654: if (i==0 && tw!=0) { foundname=1; }
655: if (i==1 && tw!=0) { foundtype=1; }
656: if (i==2 && tw!=0) { foundstat=1; }
657: if (i==3 && tw!=0) { foundend=1; }
658: }
659: if (foundstart==0 && foundend==0 && foundtype==0 && foundname==0) {
660: alert('$error1');
661: return;
662: }
663: vf.submit();
664: }
665: function flip(vf,tf) {
666: }
667: ENDPICK
668: }
669:
670: sub csvupload_javascript_forward_associate {
671: my $error1=&mt('You need to specify the name, starttime, endtime and a type');
672: return(<<ENDPICK);
673: function verify(vf) {
674: var foundstart=0;
675: var foundend=0;
676: var foundname=0;
677: var foundtype=0;
678: for (i=0;i<=vf.nfields.value;i++) {
679: tw=eval('vf.f'+i+'.selectedIndex');
680: if (tw==1) { foundname=1; }
681: if (tw==2) { foundtype=1; }
682: if (tw==3) { foundstat=1; }
683: if (tw==4) { foundend=1; }
684: }
685: if (foundstart==0 && foundend==0 && foundtype==0 && foundname==0) {
686: alert('$error1');
687: return;
688: }
689: vf.submit();
690: }
691: function flip(vf,tf) {
692: }
693: ENDPICK
694: }
695:
696: sub csv_upload_map {
1.19 albertel 697: my ($r)= @_;
1.14 albertel 698:
699: my $datatoken;
700: if (!$env{'form.datatoken'}) {
701: $datatoken=&Apache::loncommon::upfile_store($r);
702: } else {
703: $datatoken=$env{'form.datatoken'};
704: &Apache::loncommon::load_tmp_file($r);
705: }
706: my @records=&Apache::loncommon::upfile_record_sep();
707: if ($env{'form.noFirstLine'}) { shift(@records); }
1.19 albertel 708: &csvuploadmap_header($r,$datatoken,$#records+1);
1.14 albertel 709: my ($i,$keyfields);
710: if (@records) {
711: my @fields=&csvupload_fields();
712:
713: if ($env{'form.upfile_associate'} eq 'reverse') {
714: &Apache::loncommon::csv_print_samples($r,\@records);
715: $i=&Apache::loncommon::csv_print_select_table($r,\@records,
716: \@fields);
717: foreach (@fields) { $keyfields.=$_->[0].','; }
718: chop($keyfields);
719: } else {
720: unshift(@fields,['none','']);
721: $i=&Apache::loncommon::csv_samples_select_table($r,\@records,
722: \@fields);
723: my %sone=&Apache::loncommon::record_sep($records[0]);
724: $keyfields=join(',',sort(keys(%sone)));
725: }
726: }
727: &csvuploadmap_footer($r,$i,$keyfields);
728:
729: return '';
730: }
731:
732: sub csvupload_fields {
733: return (['name','Slot name'],
734: ['type','Type of slot'],
735: ['starttime','Start Time of slot'],
736: ['endtime','End Time of slot'],
1.15 albertel 737: ['startreserve','Reservation Start Time'],
1.14 albertel 738: ['ip','IP or DNS restriction'],
739: ['proctor','List of proctor ids'],
740: ['description','Slot Description'],
741: ['maxspace','Maximum number of reservations'],
742: ['symb','Resource Restriction'],
743: ['uniqueperiod','Date range of slot exclusion'],
744: ['secret','Secret word proctor uses to validate']);
745: }
746:
747: sub csv_upload_assign {
1.19 albertel 748: my ($r,$mgr)= @_;
1.14 albertel 749: &Apache::loncommon::load_tmp_file($r);
750: my @slotdata = &Apache::loncommon::upfile_record_sep();
751: if ($env{'form.noFirstLine'}) { shift(@slotdata); }
752: my %fields=&Apache::grades::get_fields();
753: $r->print('<h3>Creating Slots</h3>');
754: my $cname=$env{'course.'.$env{'request.course.id'}.'.num'};
755: my $cdom=$env{'course.'.$env{'request.course.id'}.'.domain'};
756: my $countdone=0;
757: foreach my $slot (@slotdata) {
758: my %slot;
759: my %entries=&Apache::loncommon::record_sep($slot);
760: my $domain;
761: my $name=$entries{$fields{'name'}};
762: if ($entries{$fields{'type'}}) {
763: $slot{'type'}=$entries{$fields{'type'}};
764: } else {
765: $slot{'type'}='preassigned';
766: }
767: if ($entries{$fields{'starttime'}}) {
768: $slot{'starttime'}=&UnixDate($entries{$fields{'starttime'}},"%s");
769: }
770: if ($entries{$fields{'endtime'}}) {
1.16 albertel 771: $slot{'endtime'}=&UnixDate($entries{$fields{'endtime'}},"%s");
1.14 albertel 772: }
1.23 ! albertel 773: if ($entries{$fields{'startreserve'}}) {
! 774: $slot{'startreserve'}=
! 775: &UnixDate($entries{$fields{'startreserve'}},"%s");
! 776: }
1.14 albertel 777: foreach my $key ('ip','proctor','description','maxspace',
778: 'secret','symb') {
779: if ($entries{$fields{$key}}) {
780: $slot{$key}=$entries{$fields{$key}};
781: }
782: }
783: if ($entries{$fields{'uniqueperiod'}}) {
784: my ($start,$end)=split(',',$entries{$fields{'uniqueperiod'}});
785: my @times=(&UnixDate($start,"%s"),
786: &UnixDate($end,"%s"));
787: $slot{'uniqueperiod'}=\@times;
788: }
789:
790: &Apache::lonnet::cput('slots',{$name=>\%slot},$cdom,$cname);
791: $r->print('.');
792: $r->rflush();
793: $countdone++;
794: }
795: $r->print("<br />Created $countdone slots\n");
796: $r->print("<br />\n");
1.19 albertel 797: &show_table($r,$mgr);
1.14 albertel 798: return '';
799: }
800:
1.1 albertel 801: sub handler {
802: my $r=shift;
803:
1.8 albertel 804: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'});
1.1 albertel 805: &start_page($r);
1.12 albertel 806: my $vgr=&Apache::lonnet::allowed('vgr',$env{'request.course.id'});
1.14 albertel 807: my $mgr=&Apache::lonnet::allowed('mgr',$env{'request.course.id'});
1.8 albertel 808: if ($env{'form.command'} eq 'showslots' && $vgr eq 'F') {
1.19 albertel 809: &show_table($r,$mgr);
1.14 albertel 810: } elsif ($env{'form.command'} eq 'uploadstart' && $mgr eq 'F') {
1.19 albertel 811: &upload_start($r);
1.14 albertel 812: } elsif ($env{'form.command'} eq 'csvuploadmap' && $mgr eq 'F') {
1.19 albertel 813: &csv_upload_map($r);
1.14 albertel 814: } elsif ($env{'form.command'} eq 'csvuploadassign' && $mgr eq 'F') {
815: if ($env{'form.associate'} ne 'Reverse Association') {
1.19 albertel 816: &csv_upload_assign($r,$mgr);
1.14 albertel 817: } else {
818: if ( $env{'form.upfile_associate'} ne 'reverse' ) {
819: $env{'form.upfile_associate'} = 'reverse';
820: } else {
821: $env{'form.upfile_associate'} = 'forward';
822: }
1.19 albertel 823: &csv_upload_map($r);
1.14 albertel 824: }
1.19 albertel 825: } elsif ($env{'form.command'} eq 'editslot' && $mgr eq 'F') {
826: &show_slot_edit($r);
1.8 albertel 827: } else {
1.19 albertel 828: my $symb=&Apache::lonnet::unescape($env{'form.symb'});
829: my (undef,undef,$res)=&Apache::lonnet::decode_symb($symb);
830: if ($res !~ /\.task$/) {
831: &fail($r,'not_valid');
832: return OK;
833: }
834: $env{'request.symb'}=$symb;
1.11 albertel 835: my ($status) = &Apache::lonhomework::check_task_access('0');
836: if ($status eq 'CAN_ANSWER' ||
837: $status eq 'NEEDS_CHECKIN' ||
838: $status eq 'WAITING_FOR_GRADE') {
839: &fail($r,'not_allowed');
840: return OK;
841: }
842: if ($env{'form.requestattempt'}) {
843: &show_choices($r,$symb);
844: } elsif ($env{'form.command'} eq 'release') {
845: &release_slot($r,$symb);
846: } elsif ($env{'form.command'} eq 'get') {
847: &get_slot($r,$symb);
848: } elsif ($env{'form.command'} eq 'change') {
849: &release_slot($r,$symb,$env{'form.releaseslot'},1);
850: &get_slot($r,$symb);
851: } else {
852: $r->print("<p>Unknown command: ".$env{'form.command'}."</p>");
853: }
1.2 albertel 854: }
1.1 albertel 855: &end_page($r);
856: return OK;
857: }
1.3 albertel 858:
859: 1;
860: __END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>