Annotation of loncom/interface/slotrequest.pm, revision 1.21
1.1 albertel 1: # The LearningOnline Network with CAPA
2: # Handler for requesting to have slots added to a students record
3: #
1.21 ! albertel 4: # $Id: slotrequest.pm,v 1.20 2005/10/07 20:00:20 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.21 ! 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.'maxspace" >Max space</a></th>
! 479: <th> Scheduled Students</th>
! 480: <th>'.$linkstart.'unique" >Unique Period</a></th>
1.10 albertel 481: </tr>');
1.21 ! albertel 482: my %name_cache;
! 483: my $slotsort = sub {
! 484: if ($env{'form.order'}=~/^(type|name|description|endtime|maxspace)$/) {
! 485: if (lc($slots{$a}->{$env{'form.order'}})
! 486: ne lc($slots{$b}->{$env{'form.order'}})) {
! 487: return (lc($slots{$a}->{$env{'form.order'}})
! 488: cmp lc($slots{$b}->{$env{'form.order'}}));
! 489: }
! 490: } elsif ($env{'form.order'} eq 'unique') {
! 491:
! 492: if ($slots{$a}->{'uniqueperiod'}[0]
! 493: ne $slots{$b}->{'uniqueperiod'}[0]) {
! 494: return ($slots{$a}->{'uniqueperiod'}[0]
! 495: cmp $slots{$b}->{'uniqueperiod'}[0]);
! 496: }
! 497: if ($slots{$a}->{'uniqueperiod'}[1]
! 498: ne $slots{$b}->{'uniqueperiod'}[1]) {
! 499: return ($slots{$a}->{'uniqueperiod'}[1]
! 500: cmp $slots{$b}->{'uniqueperiod'}[1]);
! 501: }
! 502: }
! 503: return $slots{$a}->{'starttime'} <=> $slots{$b}->{'starttime'};
! 504: };
! 505: foreach my $slot (sort $slotsort (keys(%slots))) {
1.5 albertel 506: if (defined($slots{$slot}->{'type'})
507: && $slots{$slot}->{'type'} ne 'schedulable_student') {
1.13 albertel 508: #next;
1.5 albertel 509: }
510: my $description=&get_description($slot,$slots{$slot});
511: my %consumed=&Apache::lonnet::dump('slot_reservations',$cdom,$cnum,
512: "^$slot\0");
513: my $ids;
514: foreach my $entry (sort(keys(%consumed))) {
515: my (undef,$id)=split("\0",$entry);
516: $ids.= $id.'-> '.$consumed{$entry}->{'name'}.'<br />';
517: }
518: my $start=localtime($slots{$slot}->{'starttime'});
519: my $end=localtime($slots{$slot}->{'endtime'});
1.14 albertel 520: my $unique;
521: if (ref($slots{$slot}{'uniqueperiod'})) {
522: $unique=localtime($slots{$slot}{'uniqueperiod'}[0]).','.
523: localtime($slots{$slot}{'uniqueperiod'}[1]);
524: }
1.20 albertel 525: my @proctors = map {
526: my ($uname,$udom)=split(/@/,$_);
1.21 ! albertel 527: my $fullname=$name_cache{$_};
! 528: if (!defined($fullname)) {
! 529: &Apache::lonnet::logthis("Gettign $uname $udom");
! 530: $fullname = &Apache::loncommon::plainname($uname,$udom);
! 531: $fullname =~s/\s/ /g;
! 532: $name_cache{$_} = $fullname;
! 533: }
1.20 albertel 534: &Apache::loncommon::aboutmewrapper($fullname,$uname,$udom);
535: } (split(/\s*,\s*/,$slots{$slot}->{'proctor'}));
536:
537: my $proctors=join(', ',@proctors);
1.14 albertel 538:
1.19 albertel 539: my $edit=(<<EDITFORM);
540: <form method="POST">
541: <input type="hidden" name="command" value="editslot" />
542: <input type="hidden" name="slot" value="$slot" />
543: <input type="submit" name="Edit" value="Edit" />
544: </form>
545: EDITFORM
1.5 albertel 546: $r->print(<<STUFF);
547: <tr>
1.21 ! albertel 548: <td rowspan="2">$edit</td>
1.5 albertel 549: <td>$slot</td>
1.10 albertel 550: <td>$slots{$slot}->{'type'}</td>
1.5 albertel 551: <td>$description</td>
552: <td>$start</td>
553: <td>$end</td>
554: <td>$slots{$slot}->{'maxspace'}</td>
555: <td>$ids</td>
1.14 albertel 556: <td>$unique</td>
1.5 albertel 557: </tr>
1.21 ! albertel 558: <tr>
! 559: <td colspan="8">$proctors</td>
! 560: </tr>
1.5 albertel 561: STUFF
562: }
563: $r->print('</table>');
564: }
565:
1.14 albertel 566: sub upload_start {
1.19 albertel 567: my ($r)=@_;
1.14 albertel 568: $r->print(&Apache::grades::checkforfile_js());
569: my $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";
570: $result.=' <b>'.
571: &mt('Specify a file containing the slot definitions.').
572: '</b></td></tr>'."\n";
573: $result.='<tr bgcolor=#ffffe6><td>'."\n";
574: my $upfile_select=&Apache::loncommon::upfile_select_html();
575: my $ignore=&mt('Ignore First Line');
576: $result.=<<ENDUPFORM;
577: <form method="post" enctype="multipart/form-data" action="/adm/slotrequest" name="slotupload">
578: <input type="hidden" name="command" value="csvuploadmap" />
579: $upfile_select
580: <br /><input type="button" onClick="javascript:checkUpload(this.form);" value="Upload Data" />
581: <label><input type="checkbox" name="noFirstLine" />$ignore</label>
582: </form>
583: ENDUPFORM
584: $result.='</td></tr></table>'."\n";
585: $result.='</td></tr></table>'."\n";
586: $r->print($result);
587: }
588:
589: sub csvuploadmap_header {
1.19 albertel 590: my ($r,$datatoken,$distotal)= @_;
1.14 albertel 591: my $javascript;
592: if ($env{'form.upfile_associate'} eq 'reverse') {
593: $javascript=&csvupload_javascript_reverse_associate();
594: } else {
595: $javascript=&csvupload_javascript_forward_associate();
596: }
597:
598: my $checked=(($env{'form.noFirstLine'})?' checked="checked"':'');
599: my $ignore=&mt('Ignore First Line');
600: $r->print(<<ENDPICK);
601: <form method="post" enctype="multipart/form-data" action="/adm/slotrequest" name="slotupload">
602: <h3>Identify fields</h3>
603: Total number of records found in file: $distotal <hr />
604: Enter as many fields as you can. The system will inform you and bring you back
605: to this page if the data selected is insufficient to create the slots.<hr />
606: <input type="button" value="Reverse Association" onClick="javascript:this.form.associate.value='Reverse Association';submit(this.form);" />
607: <label><input type="checkbox" name="noFirstLine" $checked />$ignore</label>
608: <input type="hidden" name="associate" value="" />
609: <input type="hidden" name="datatoken" value="$datatoken" />
610: <input type="hidden" name="fileupload" value="$env{'form.fileupload'}" />
611: <input type="hidden" name="upfiletype" value="$env{'form.upfiletype'}" />
612: <input type="hidden" name="upfile_associate"
613: value="$env{'form.upfile_associate'}" />
614: <input type="hidden" name="command" value="csvuploadassign" />
615: <hr />
616: <script type="text/javascript" language="Javascript">
617: $javascript
618: </script>
619: ENDPICK
620: return '';
621:
622: }
623:
624: sub csvuploadmap_footer {
625: my ($request,$i,$keyfields) =@_;
626: $request->print(<<ENDPICK);
627: </table>
628: <input type="hidden" name="nfields" value="$i" />
629: <input type="hidden" name="keyfields" value="$keyfields" />
630: <input type="button" onClick="javascript:verify(this.form)" value="Create Slots" /><br />
631: </form>
632: ENDPICK
633: }
634:
635: sub csvupload_javascript_reverse_associate {
636: my $error1=&mt('You need to specify the name, starttime, endtime and a type');
637: return(<<ENDPICK);
638: function verify(vf) {
639: var foundstart=0;
640: var foundend=0;
641: var foundname=0;
642: var foundtype=0;
643: for (i=0;i<=vf.nfields.value;i++) {
644: tw=eval('vf.f'+i+'.selectedIndex');
645: if (i==0 && tw!=0) { foundname=1; }
646: if (i==1 && tw!=0) { foundtype=1; }
647: if (i==2 && tw!=0) { foundstat=1; }
648: if (i==3 && tw!=0) { foundend=1; }
649: }
650: if (foundstart==0 && foundend==0 && foundtype==0 && foundname==0) {
651: alert('$error1');
652: return;
653: }
654: vf.submit();
655: }
656: function flip(vf,tf) {
657: }
658: ENDPICK
659: }
660:
661: sub csvupload_javascript_forward_associate {
662: my $error1=&mt('You need to specify the name, starttime, endtime and a type');
663: return(<<ENDPICK);
664: function verify(vf) {
665: var foundstart=0;
666: var foundend=0;
667: var foundname=0;
668: var foundtype=0;
669: for (i=0;i<=vf.nfields.value;i++) {
670: tw=eval('vf.f'+i+'.selectedIndex');
671: if (tw==1) { foundname=1; }
672: if (tw==2) { foundtype=1; }
673: if (tw==3) { foundstat=1; }
674: if (tw==4) { foundend=1; }
675: }
676: if (foundstart==0 && foundend==0 && foundtype==0 && foundname==0) {
677: alert('$error1');
678: return;
679: }
680: vf.submit();
681: }
682: function flip(vf,tf) {
683: }
684: ENDPICK
685: }
686:
687: sub csv_upload_map {
1.19 albertel 688: my ($r)= @_;
1.14 albertel 689:
690: my $datatoken;
691: if (!$env{'form.datatoken'}) {
692: $datatoken=&Apache::loncommon::upfile_store($r);
693: } else {
694: $datatoken=$env{'form.datatoken'};
695: &Apache::loncommon::load_tmp_file($r);
696: }
697: my @records=&Apache::loncommon::upfile_record_sep();
698: if ($env{'form.noFirstLine'}) { shift(@records); }
1.19 albertel 699: &csvuploadmap_header($r,$datatoken,$#records+1);
1.14 albertel 700: my ($i,$keyfields);
701: if (@records) {
702: my @fields=&csvupload_fields();
703:
704: if ($env{'form.upfile_associate'} eq 'reverse') {
705: &Apache::loncommon::csv_print_samples($r,\@records);
706: $i=&Apache::loncommon::csv_print_select_table($r,\@records,
707: \@fields);
708: foreach (@fields) { $keyfields.=$_->[0].','; }
709: chop($keyfields);
710: } else {
711: unshift(@fields,['none','']);
712: $i=&Apache::loncommon::csv_samples_select_table($r,\@records,
713: \@fields);
714: my %sone=&Apache::loncommon::record_sep($records[0]);
715: $keyfields=join(',',sort(keys(%sone)));
716: }
717: }
718: &csvuploadmap_footer($r,$i,$keyfields);
719:
720: return '';
721: }
722:
723: sub csvupload_fields {
724: return (['name','Slot name'],
725: ['type','Type of slot'],
726: ['starttime','Start Time of slot'],
727: ['endtime','End Time of slot'],
1.15 albertel 728: ['startreserve','Reservation Start Time'],
1.14 albertel 729: ['ip','IP or DNS restriction'],
730: ['proctor','List of proctor ids'],
731: ['description','Slot Description'],
732: ['maxspace','Maximum number of reservations'],
733: ['symb','Resource Restriction'],
734: ['uniqueperiod','Date range of slot exclusion'],
735: ['secret','Secret word proctor uses to validate']);
736: }
737:
738: sub csv_upload_assign {
1.19 albertel 739: my ($r,$mgr)= @_;
1.14 albertel 740: &Apache::loncommon::load_tmp_file($r);
741: my @slotdata = &Apache::loncommon::upfile_record_sep();
742: if ($env{'form.noFirstLine'}) { shift(@slotdata); }
743: my %fields=&Apache::grades::get_fields();
744: $r->print('<h3>Creating Slots</h3>');
745: my $cname=$env{'course.'.$env{'request.course.id'}.'.num'};
746: my $cdom=$env{'course.'.$env{'request.course.id'}.'.domain'};
747: my $countdone=0;
748: foreach my $slot (@slotdata) {
749: my %slot;
750: my %entries=&Apache::loncommon::record_sep($slot);
751: my $domain;
752: my $name=$entries{$fields{'name'}};
753: if ($entries{$fields{'type'}}) {
754: $slot{'type'}=$entries{$fields{'type'}};
755: } else {
756: $slot{'type'}='preassigned';
757: }
758: if ($entries{$fields{'starttime'}}) {
759: $slot{'starttime'}=&UnixDate($entries{$fields{'starttime'}},"%s");
760: }
761: if ($entries{$fields{'endtime'}}) {
1.16 albertel 762: $slot{'endtime'}=&UnixDate($entries{$fields{'endtime'}},"%s");
1.14 albertel 763: }
764: foreach my $key ('ip','proctor','description','maxspace',
765: 'secret','symb') {
766: if ($entries{$fields{$key}}) {
767: $slot{$key}=$entries{$fields{$key}};
768: }
769: }
770: if ($entries{$fields{'uniqueperiod'}}) {
771: my ($start,$end)=split(',',$entries{$fields{'uniqueperiod'}});
772: my @times=(&UnixDate($start,"%s"),
773: &UnixDate($end,"%s"));
774: $slot{'uniqueperiod'}=\@times;
775: }
776:
777: &Apache::lonnet::cput('slots',{$name=>\%slot},$cdom,$cname);
778: $r->print('.');
779: $r->rflush();
780: $countdone++;
781: }
782: $r->print("<br />Created $countdone slots\n");
783: $r->print("<br />\n");
1.19 albertel 784: &show_table($r,$mgr);
1.14 albertel 785: return '';
786: }
787:
1.1 albertel 788: sub handler {
789: my $r=shift;
790:
1.8 albertel 791: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'});
1.1 albertel 792: &start_page($r);
1.12 albertel 793: my $vgr=&Apache::lonnet::allowed('vgr',$env{'request.course.id'});
1.14 albertel 794: my $mgr=&Apache::lonnet::allowed('mgr',$env{'request.course.id'});
1.8 albertel 795: if ($env{'form.command'} eq 'showslots' && $vgr eq 'F') {
1.19 albertel 796: &show_table($r,$mgr);
1.14 albertel 797: } elsif ($env{'form.command'} eq 'uploadstart' && $mgr eq 'F') {
1.19 albertel 798: &upload_start($r);
1.14 albertel 799: } elsif ($env{'form.command'} eq 'csvuploadmap' && $mgr eq 'F') {
1.19 albertel 800: &csv_upload_map($r);
1.14 albertel 801: } elsif ($env{'form.command'} eq 'csvuploadassign' && $mgr eq 'F') {
802: if ($env{'form.associate'} ne 'Reverse Association') {
1.19 albertel 803: &csv_upload_assign($r,$mgr);
1.14 albertel 804: } else {
805: if ( $env{'form.upfile_associate'} ne 'reverse' ) {
806: $env{'form.upfile_associate'} = 'reverse';
807: } else {
808: $env{'form.upfile_associate'} = 'forward';
809: }
1.19 albertel 810: &csv_upload_map($r);
1.14 albertel 811: }
1.19 albertel 812: } elsif ($env{'form.command'} eq 'editslot' && $mgr eq 'F') {
813: &show_slot_edit($r);
1.8 albertel 814: } else {
1.19 albertel 815: my $symb=&Apache::lonnet::unescape($env{'form.symb'});
816: my (undef,undef,$res)=&Apache::lonnet::decode_symb($symb);
817: if ($res !~ /\.task$/) {
818: &fail($r,'not_valid');
819: return OK;
820: }
821: $env{'request.symb'}=$symb;
1.11 albertel 822: my ($status) = &Apache::lonhomework::check_task_access('0');
823: if ($status eq 'CAN_ANSWER' ||
824: $status eq 'NEEDS_CHECKIN' ||
825: $status eq 'WAITING_FOR_GRADE') {
826: &fail($r,'not_allowed');
827: return OK;
828: }
829: if ($env{'form.requestattempt'}) {
830: &show_choices($r,$symb);
831: } elsif ($env{'form.command'} eq 'release') {
832: &release_slot($r,$symb);
833: } elsif ($env{'form.command'} eq 'get') {
834: &get_slot($r,$symb);
835: } elsif ($env{'form.command'} eq 'change') {
836: &release_slot($r,$symb,$env{'form.releaseslot'},1);
837: &get_slot($r,$symb);
838: } else {
839: $r->print("<p>Unknown command: ".$env{'form.command'}."</p>");
840: }
1.2 albertel 841: }
1.1 albertel 842: &end_page($r);
843: return OK;
844: }
1.3 albertel 845:
846: 1;
847: __END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>