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