Annotation of loncom/interface/londropadd.pm, revision 1.8
1.1 www 1: # The LearningOnline Network with CAPA
2: # Handler to drop and add students in courses
3: #
4: # (Handler to set parameters for assessments
5: #
6: # (Handler to resolve ambiguous file locations
7: #
8: # (TeX Content Handler
9: #
10: # 05/29/00,05/30,10/11 Gerd Kortemeyer)
11: #
12: # 10/11,10/12,10/16 Gerd Kortemeyer)
13: #
14: # 11/20,11/21,11/22,11/23,11/24,11/25,11/27,11/28,
15: # 12/08,12/12 Gerd Kortemeyer)
16: #
1.7 www 17: # 12/26,12/27,12/28,
18: # 01/01/01 Gerd Kortemeyer
1.1 www 19:
20: package Apache::londropadd;
21:
22: use strict;
23: use Apache::lonnet;
24: use Apache::Constants qw(:common :http REDIRECT);
25:
26:
27: # ================================================================ Main Handler
28:
29: sub handler {
30: my $r=shift;
31:
32: if ($r->header_only) {
33: $r->content_type('text/html');
34: $r->send_http_header;
35: return OK;
36: }
37:
38: # ----------------------------------------------------- Needs to be in a course
39:
40: if (($ENV{'request.course.fn'}) &&
41: (&Apache::lonnet::allowed('cst',$ENV{'request.course.id'}))) {
42:
43: # ------------------------------------------------------------------ Start page
44: $r->content_type('text/html');
45: $r->send_http_header;
46: $r->print(<<ENDHEAD);
47: <html>
48: <head>
49: <title>LON-CAPA Student Drop/Add</title>
50: </head>
51: <body bgcolor="#FFFFFF">
52: <img align=right src=/adm/lonIcons/lonlogos.gif>
53: <h1>Drop/Add Students</h1>
54: <form method="post" enctype="multipart/form-data"
55: action="/adm/dropadd" name="studentform">
56: <h2>Course: $ENV{'course.'.$ENV{'request.course.id'}.'.description'}</h2>
57: ENDHEAD
1.2 www 58: # --------------------------------------------------- Phase one, initial screen
59: unless ($ENV{'form.phase'}) {
60: $r->print(<<ENDUPFORM);
61: <input type=hidden name=phase value=two>
62: <hr>
63: <h3>Upload a courselist</h3>
64: <input type=file name=upfile size=50>
65: <br>Type: <select name=upfiletype>
66: <option value=csv>CSV (comma separated values, spreadsheet)</option>
67: <option value=space>Space separated</option>
68: <option value=tab>Tabulator separated</option>
69: <option value=xml>HTML/XML</option>
70: </select>
71: <p><input type=submit name=fileupload value="Upload Courselist">
72: <hr>
73: <h3>Enroll a single student</h3>
74: <p><input type=submit name=enroll value="Enroll Student">
75: <hr>
76: <h3>Drop a student</h3>
77: <p><input type=submit name=drop value="Drop Student">
78: ENDUPFORM
79: }
80: # ------------------------------------------------------------------- Phase two
81: if ($ENV{'form.phase'} eq 'two') {
82: if ($ENV{'form.fileupload'}) {
1.4 www 83: my $datatoken=$ENV{'user.name'}.'_'.$ENV{'user.domain'}.
84: '_enroll_'.$ENV{'request.course.id'}.'_'.time.'_'.$$;
85: {
86: my $fh=Apache::File->new('>'.$r->dir_config('lonDaemons').
87: '/tmp/'.$datatoken.'.tmp');
88: print $fh $ENV{'form.upfile'};
89: }
1.2 www 90: my $separator='';
91: my $remove='';
92: if ($ENV{'form.upfiletype'} eq 'csv') {
93: $separator='\"\,\s*\"';
94: $remove='"';
95: } elsif ($ENV{'form.upfiletype'} eq 'space') {
96: $separator='\s+';
97: } elsif ($ENV{'form.upfiletype'} eq 'tab') {
98: $separator='\t+';
99: } elsif ($ENV{'form.upfiletype'} eq 'xml') {
100: }
101: my @lines=split(/\n/,$ENV{'form.upfile'});
1.4 www 102: my $total=$#lines;
1.3 www 103: $ENV{'SERVER_NAME'}=~/(\w+\.\w+)$/;
104: my $krbdefdom=$1;
105: $krbdefdom=~tr/a-z/A-Z/;
1.4 www 106: my $today=time;
107: my $halfyear=$today+15552000;
108: my $defdom=$r->dir_config('lonDefDomain');
1.2 www 109: $r->print(<<ENDPICK);
110: <input type=hidden name=phase value=three>
111: <input type=hidden name=datatoken value="$datatoken">
112: <input type=hidden name=upfiletype value=$ENV{'form.upfiletype'}>
113: <hr>
114: <h3>Identify fields</h3>
1.4 www 115: Total number of records found in file: $total
1.3 www 116: <script>
117: function verify(vf) {
118: var founduname=0;
119: var foundpwd=0;
120: var foundname=0;
121: var foundid=0;
122: var foundsec=0;
123: var foundatype=0;
124: var tw;
125: var message='';
126: for (i=0;i<=vf.nfields.value;i++) {
127: tw=eval('vf.f'+i+'.selectedIndex');
128: if (tw==1) { founduname=1; }
129: if ((tw>=2) && (tw<=6)) { foundname=1; }
130: if (tw==7) { foundid=1; }
131: if (tw==8) { foundsec=1; }
132: if (tw==9) { foundpwd=1; }
133: }
134: if (founduname==0) {
135: alert('You need to specify at least the username field');
136: return;
137: }
138: if (vf.login[0].checked) {
139: foundatype=1;
140: if (vf.krbdom.value=='') {
141: alert('You need to specify the Kerberos domain');
142: return;
143: }
144: }
145: if (vf.login[1].checked) {
146: foundatype=1;
147: if ((vf.intpwd.value=='') && (foundpwd==0)) {
148: alert('You need to specify the initial password');
149: return;
150: }
151: }
152: if (foundatype==0) {
153: alert('You need to set the login type');
154: return;
155: }
156: if (foundname==0) { message='No name fields specified. '; }
157: if (foundid==0) { message+='No ID or student number field specified. '; }
158: if (foundsec==0) { message+='No section or group field specified. '; }
1.4 www 159: if (vf.startdate.value=='') {
160: message+='No starting date set. ';
1.3 www 161: }
1.4 www 162: if (vf.enddate.value=='') {
163: message+='No ending date set. ';
164: }
165: if ((vf.enddate.value!='') && (vf.startdate.value!='')) {
166: if (vf.enddate.value<vf.startdate.value) {
167: alert('Ending date is before starting date');
168: return;
169: }
170: }
171: if (message!='') {
172: message+='Continue enrollment?';
173: if (confirm(message)) {
174: pclose();
175: vf.submit();
176: }
177: } else {
178: pclose();
179: vf.submit();
180: }
1.3 www 181: }
182:
183: function flip(vf,tf) {
184: var nw=eval('vf.f'+tf+'.selectedIndex');
185: var i;
186: for (i=0;i<=vf.nfields.value;i++) {
187: if ((i!=tf) && (eval('vf.f'+i+'.selectedIndex')==nw)) {
188: eval('vf.f'+i+'.selectedIndex=0;')
189: }
190: }
191: if (nw==2) {
192: for (i=0;i<=vf.nfields.value;i++) {
193: if ((eval('vf.f'+i+'.selectedIndex')>=3) &&
194: (eval('vf.f'+i+'.selectedIndex')<=6)) {
195: eval('vf.f'+i+'.selectedIndex=0;')
196: }
197: }
198: }
199: if ((nw>=3) && (nw<=6)) {
200: for (i=0;i<=vf.nfields.value;i++) {
201: if (eval('vf.f'+i+'.selectedIndex')==2) {
202: eval('vf.f'+i+'.selectedIndex=0;')
203: }
204: }
205: }
206: if (nw==9) {
207: vf.login[1].checked=true;
208: vf.intpwd.value='';
209: vf.krbdom.value='';
210: }
211:
212: }
213:
214: function clearpwd(vf) {
215: var i;
216: for (i=0;i<=vf.nfields.value;i++) {
217: if (eval('vf.f'+i+'.selectedIndex')==9) {
218: eval('vf.f'+i+'.selectedIndex=0;')
219: }
220: }
221: }
222:
223: function setkrb(vf) {
224: if (vf.krbdom.value!='') {
225: clearpwd(vf);
226: vf.login[0].checked=true;
227: vf.krbdom.value=vf.krbdom.value.toUpperCase();
228: vf.intpwd.value='';
229: }
230: }
231:
232: function setint(vf) {
233: if (vf.intpwd.value!='') {
234: clearpwd(vf);
235: vf.login[1].checked=true;
236: vf.krbdom.value='';
237: }
238: }
239:
240: function clickkrb(vf) {
241: vf.krbdom.value='$krbdefdom';
242: clearpwd(vf);
243: vf.intpwd.value='';
244: }
245:
246: function clickint(vf) {
247: vf.krbdom.value='';
248: }
249:
1.4 www 250: function pclose() {
251: parmwin=window.open("/adm/rat/empty.html","LONCAPAparms",
252: "height=350,width=350,scrollbars=no,menubar=no");
253: parmwin.close();
254: }
255:
256: function pjump(type,dis,value,marker,ret,call) {
257: parmwin=window.open("/adm/rat/parameter.html?type="+escape(type)
258: +"&value="+escape(value)+"&marker="+escape(marker)
259: +"&return="+escape(ret)
260: +"&call="+escape(call)+"&name="+escape(dis),"LONCAPAparms",
261: "height=350,width=350,scrollbars=no,menubar=no");
262:
263: }
264:
265: function dateset() {
266: if (document.studentform.pres_marker.value=='end') {
267: document.studentform.enddate.value=
268: document.studentform.pres_value.value;
269: }
270: if (document.studentform.pres_marker.value=='start') {
271: document.studentform.startdate.value=
272: document.studentform.pres_value.value;
273: }
274: pclose();
275: }
1.3 www 276:
277: </script>
1.2 www 278: <table border=2><tr><th>Field</th><th>Samples</th></tr>
279: ENDPICK
280: my @sone; my @stwo; my @sthree; my $nfields=0;
281: if ($#lines>=0) {
282: $lines[0]=~s/^$remove//;
283: $lines[0]=~s/$remove$//;
284: @sone=split(/$separator/,$lines[0]);
285: $nfields=$#sone;
286: if ($#lines>=1) {
287: $lines[1]=~s/^$remove//;
288: $lines[1]=~s/$remove$//;
289: @stwo=split(/$separator/,$lines[1]);
290: $nfields=$#stwo;
291: }
292: if ($#lines>=2) {
293: $lines[2]=~s/^$remove//;
294: $lines[2]=~s/$remove$//;
295: @sthree=split(/$separator/,$lines[2]);
296: $nfields=$#sthree;
297: }
298: my $i;
299: for ($i=0;$i<=$nfields;$i++) {
1.3 www 300: $r->print('<tr><td><select name=f'.$i.
301: ' onChange="flip(this.form,'.$i.');">');
1.2 www 302: map {
303: my ($value,$display)=split(/\:/,$_);
304: $r->print('<option value='.$value.'>'.$display.
305: '</option>');
306: } ('none: ','username:Username',
307: 'names:Last Name, First Names',
308: 'fname:First Name','mname:Middle Names/Initials',
309: 'lname:Last Name','gen:Generation',
1.3 www 310: 'id:ID/Student Number','sec:Group/Section',
311: 'ipwd:Initial Password');
1.2 www 312: $r->print('</select></td><td>');
313: if (defined($sone[$i])) {
314: $r->print($sone[$i]."</br>\n");
315: }
316: if (defined($stwo[$i])) {
317: $r->print($stwo[$i]."</br>\n");
318: }
319: if (defined($sthree[$i])) {
320: $r->print($sthree[$i]."</br>\n");
321: }
322: $r->print('</td></tr>');
323: }
324: }
1.3 www 325: $r->print(<<ENDPICK);
326: </table>
327: <input type=hidden name=nfields value=$nfields>
328: <h3>Login Type</h3>
329: <input type=radio name=login value=krb onClick="clickkrb(this.form);">
330: Kerberos authenticated with domain
331: <input type=text size=10 name=krbdom onChange="setkrb(this.form);"><p>
332: <input type=radio name=login value=int onClick="clickint(this.form);">
333: Internally authenticated (with initial password
1.5 www 334: <input type=text size=10 name=intpwd onChange="setint(this.form);">)
335: <h3>LON-CAPA Domain for Students</h3>
1.4 www 336: LON-CAPA domain: <input type=text size=10 value=$defdom name=lcdomain><p>
1.5 www 337: <h3>Starting and Ending Dates</h3>
1.4 www 338: <input type="hidden" value='' name="pres_value">
339: <input type="hidden" value='' name="pres_type">
340: <input type="hidden" value='' name="pres_marker">
341: <input type="hidden" value='$today' name=startdate>
342: <input type="hidden" value='$halfyear' name=enddate>
343: <a
344: href="javascript:pjump('date_start','Enrollment Starting Date',document.studentform.startdate.value,'start','studentform.pres','dateset');"
345: >Set Starting Date</a><p>
346:
347: <a
348: href="javascript:pjump('date_end','Enrollment Ending Date',document.studentform.enddate.value,'end','studentform.pres','dateset');"
349: >Set Ending Date</a><p>
1.5 www 350: <h3>Full Update</h3>
351: <input type=checkbox name=fullup value=yes> Full update
352: (also dropping students)<p>
1.6 www 353: <input type=button onClick="verify(this.form)" value="Update Courselist"><br>
354: Note: for large courses, this operation might be time consuming.
1.3 www 355: ENDPICK
1.2 www 356: } elsif ($ENV{'form.enroll'}) {
357: } elsif ($ENV{'form.drop'}) {
358: }
359: }
360: # ----------------------------------------------------------------- Phase three
361: if ($ENV{'form.phase'} eq 'three') {
362: if ($ENV{'form.datatoken'}) {
363: my $separator='';
364: my $remove='';
365: if ($ENV{'form.upfiletype'} eq 'csv') {
366: $separator='\"\,\s*\"';
367: $remove='"';
368: } elsif ($ENV{'form.upfiletype'} eq 'space') {
369: $separator='\s+';
370: } elsif ($ENV{'form.upfiletype'} eq 'tab') {
371: $separator='\t+';
372: } elsif ($ENV{'form.upfiletype'} eq 'xml') {
373: }
1.4 www 374: my %fields=();
375: for (my $i=0;$i<=$ENV{'form.nfields'};$i++) {
376: $fields{$ENV{'form.f'.$i}}=$i;
377: }
378: my $startdate=$ENV{'form.startdate'};
379: my $enddate=$ENV{'form.enddate'};
380: if ($startdate=~/\D/) { $startdate=''; }
381: if ($enddate=~/\D/) { $enddate=''; }
1.5 www 382: my $domain=$ENV{'form.lcdomain'};
383: my $amode='';
384: my $genpwd='';
385: if ($ENV{'form.login'} eq 'krb') {
386: $amode='krb4';
387: $genpwd=$ENV{'form.krbdom'};
388: } elsif ($ENV{'form.login'} eq 'int') {
389: $amode='internal';
390: if ((defined($ENV{'form.intpwd'})) && ($ENV{'form.intpwd'})) {
391: $genpwd=$ENV{'form.intpwd'};
392: }
393: }
394: unless (($domain=~/\W/) || ($amode eq '')) {
395: $r->print('<h3>Enrolling Students</h3>');
1.7 www 396: my $count=0;
397: my $flushc=0;
398: my %student=();
1.5 www 399: # ----------------------------------------------------------- Get new classlist
400: my @studentdata=();
1.4 www 401: {
402: my $fh;
403: if ($fh=Apache::File->new($r->dir_config('lonDaemons').
404: '/tmp/'.$ENV{'form.datatoken'}.'.tmp')) {
405: @studentdata=<$fh>;
406: }
407: }
1.5 www 408: # --------------------------------------------------------- Enroll new students
1.2 www 409: map {
1.4 www 410: my $line=$_;
411: chomp($line);
412: $line=~s/^$remove//;
413: $line=~s/$remove$//;
414: my @entries=split(/$separator/,$line);
415: unless (($entries[$fields{'username'}] eq '') ||
416: (!defined($entries[$fields{'username'}]))) {
417: my $fname=''; my $mname=''; my $lname=''; my $gen='';
418: if (defined($fields{'names'})) {
419: ($lname,$fname,$mname)=
420: ($entries[$fields{'names'}]=~/([^\,]+)\,\s*(\w+)\s*(.*)$/);
421: } else {
422: if (defined($fields{'fname'})) {
423: $fname=$entries[$fields{'fname'}];
424: }
425: if (defined($fields{'mname'})) {
426: $mname=$entries[$fields{'mname'}];
427: }
428: if (defined($fields{'lname'})) {
429: $lname=$entries[$fields{'lname'}];
430: }
431: if (defined($fields{'gen'})) {
432: $gen=$entries[$fields{'gen'}];
433: }
434: }
435: if ($entries[$fields{'username'}]=~/\W/) {
436: $r->print('<p><b>Unacceptable username: '.
437: $entries[$fields{'username'}].' for user '.
438: $fname.' '.$mname.' '.$lname.' '.$gen.'</b><p>');
1.5 www 439: } else {
440: my $sec='';
441: my $username=$entries[$fields{'username'}];
442: if (defined($fields{'sec'})) {
443: if (defined($entries[$fields{'sec'}])) {
444: $sec=$entries[$fields{'sec'}];
445: }
446: }
447: my $id='';
448: if (defined($fields{'id'})) {
449: if (defined($entries[$fields{'id'}])) {
450: $id=$entries[$fields{'id'}];
451: }
452: $id=~tr/A-Z/a-z/;
453: }
454: my $password='';
455: if ($genpwd) {
456: $password=$genpwd;
457: } else {
458: if (defined($fields{'ipwd'})) {
459: if ($entries[$fields{'ipwd'}]) {
460: $password=$entries[$fields{'ipwd'}];
461: }
462: }
463: }
464: if ($password) {
465: my $reply=&Apache::lonnet::modifystudent(
466: $domain,$username,$id,$amode,$password,
467: $fname,$mname,$lname,$gen,$sec,$enddate,$startdate);
468: unless ($reply eq 'ok') {
469: $r->print(
470: "<p><b>Error enrolling $username: $reply</b><p>");
1.6 www 471: } else {
1.7 www 472: $count++; $flushc++;
473: $student{$username}=1;
1.6 www 474: $r->print('. ');
1.7 www 475: if ($flushc>15) {
476: $r->rflush;
477: $flushc=0;
478: }
1.6 www 479: }
1.5 www 480: } else {
481: $r->print(
482: "<p><b>No password for $username</b><p>");
483: }
1.4 www 484: }
485: }
486: } @studentdata;
1.7 www 487: $r->print('<p>Processed Students: '.$count);
1.5 www 488: # --------------------------------------------------------------- Drop students
489: if ($ENV{'form.fullup'} eq 'yes') {
490: $r->print('<h3>Dropping Students</h3>');
491: # ------------------------------------------------------- Get current classlist
492: my $cid=$ENV{'request.course.id'};
493: my $classlst=&Apache::lonnet::reply
494: ('dump:'.$ENV{'course.'.$cid.'.domain'}.':'.
495: $ENV{'course.'.$cid.'.num'}.':classlist',
496: $ENV{'course.'.$cid.'.home'});
497: my %currentlist=();
498: my $now=time;
499: unless ($classlst=~/^error\:/) {
1.7 www 500: map {
501: my ($name,$value)=split(/\=/,$_);
1.8 ! www 502: my ($end,$start)=split(/\:/,
1.7 www 503: &Apache::lonnet::unescape($value));
504: my $active=1;
505: if (($end) && ($now>$end)) { $active=0; }
506: if ($active) {
507: $currentlist{&Apache::lonnet::unescape($name)}=1;
508: }
509: } split(/\&/,$classlst);
510:
1.5 www 511: } else {
512: $r->print(
513: '<font color=red><h3>Could not access classlist: '.$classlst.
514: '</h3></font>');
515: }
516: }
517: # ------------------------------------------------------------------------ Done
1.2 www 518: }
1.5 www 519: }
1.2 www 520: }
521: # ------------------------------------------------------------------------- End
1.1 www 522: $r->print('</form></body></html>');
523: } else {
524: # ----------------------------- Not in a course, or not allowed to modify parms
525: $ENV{'user.error.msg'}=
526: "/adm/dropadd:cst:0:0:Cannot drop or add students";
527: return HTTP_NOT_ACCEPTABLE;
528: }
529: return OK;
530: }
531:
532: 1;
533: __END__
534:
535:
536:
537:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>