Annotation of rat/lonratsrv.pm, revision 1.30
1.1 www 1: # The LearningOnline Network with CAPA
2: # Server for RAT Maps
3: #
1.30 ! albertel 4: # $Id: lonratsrv.pm,v 1.29 2005/02/17 09:09:57 albertel Exp $
1.16 www 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: #
1.1 www 28:
29: package Apache::lonratsrv;
30:
31: use strict;
32: use Apache::Constants qw(:common);
1.2 www 33: use Apache::File;
34: use HTML::TokeParser;
1.30 ! albertel 35: use Apache::lonnet;
1.2 www 36:
1.4 www 37: # ------------------------------------------------------------- From RAT to XML
1.2 www 38:
39: sub qtescape {
40: my $str=shift;
1.4 www 41: $str=~s/\&\#58\;/\:/g;
42: $str=~s/\&\#39\;/\'/g;
43: $str=~s/\&\#44\;/\,/g;
1.15 www 44: $str=~s/\"/\&\#34\;/g;
1.2 www 45: return $str;
46: }
47:
1.4 www 48: # ------------------------------------------------------------- From XML to RAT
1.2 www 49:
1.4 www 50: sub qtunescape {
1.2 www 51: my $str=shift;
1.14 www 52: $str=~s/\:/\&colon\;/g;
1.4 www 53: $str=~s/\'/\&\#39\;/g;
54: $str=~s/\,/\&\#44\;/g;
55: $str=~s/\"/\&\#34\;/g;
1.2 www 56: return $str;
57: }
58:
59: # --------------------------------------------------------- Loads map from disk
60:
61: sub loadmap {
1.28 www 62: my ($fn,$errtext,$infotext)=@_;
63: if ($errtext) { return('',$errtext); }
1.2 www 64: my $outstr='';
65: my @obj=();
66: my @links=();
1.21 www 67: my $instr='';
68: if ($fn=~/^\/*uploaded\//) {
69: $instr=&Apache::lonnet::getfile($fn);
70: } elsif (-e $fn) {
71: my @content=();
1.2 www 72: {
73: my $fh=Apache::File->new($fn);
74: @content=<$fh>;
75: }
1.21 www 76: $instr=join('',@content);
77: }
1.25 albertel 78: if ($instr eq -2) {
79: $errtext.='Map not loaded: An error occured while trying to load the map.';
80: } elsif ($instr) {
1.2 www 81: my $parser = HTML::TokeParser->new(\$instr);
82: my $token;
83: my $graphmode=0;
84:
85: $fn=~/\.(\w+)$/;
86: $outstr="mode<:>$1";
87:
88: while ($token = $parser->get_token) {
89: if ($token->[0] eq 'S') {
90: if ($token->[1] eq 'map') {
91: $graphmode=($token->[2]->{'mode'} eq 'rat/graphical');
92: } elsif ($token->[1] eq 'resource') {
1.3 www 93: # -------------------------------------------------------------------- Resource
94: $outstr.='<&>objcont';
95: if ($token->[2]->{'id'}) {
96: $outstr.='<:>'.$token->[2]->{'id'};
97: if ($obj[$token->[2]->{'id'}]==1) {
98: $errtext.='Error: multiple use of ID '.
99: $token->[2]->{'id'}.'. ';
100: }
101: $obj[$token->[2]->{'id'}]=1;
102: } else {
103: my $i=1;
104: while (($i<=$#obj) && ($obj[$i]!=0)) { $i++; }
105: $outstr.='<:>'.$i;
106: $obj[$i]=1;
107: }
108: $outstr.='<:>';
1.4 www 109: $outstr.=qtunescape($token->[2]->{'title'}).":";
110: $outstr.=qtunescape($token->[2]->{'src'}).":";
1.14 www 111: if ($token->[2]->{'external'} eq 'true') {
1.4 www 112: $outstr.='true:';
113: } else {
114: $outstr.='false:';
115: }
116: if ($token->[2]->{'type'}) {
117: $outstr.=$token->[2]->{'type'}.':';
118: } else {
119: $outstr.='normal:';
120: }
121: $outstr.='res';
1.2 www 122: } elsif ($token->[1] eq 'condition') {
1.3 www 123: # ------------------------------------------------------------------- Condition
124: $outstr.='<&>objcont';
125: if ($token->[2]->{'id'}) {
126: $outstr.='<:>'.$token->[2]->{'id'};
127: if ($obj[$token->[2]->{'id'}]==1) {
128: $errtext.='Error: multiple use of ID '.
129: $token->[2]->{'id'}.'. ';
130: }
131: $obj[$token->[2]->{'id'}]=1;
132: } else {
133: my $i=1;
134: while (($i<=$#obj) && ($obj[$i]!=0)) { $i++; }
135: $outstr.='<:>'.$i;
136: $obj[$i]=1;
137: }
138: $outstr.='<:>';
1.4 www 139: $outstr.=qtunescape($token->[2]->{'value'}).':';
140: if ($token->[2]->{'type'}) {
141: $outstr.=$token->[2]->{'type'}.':';
142: } else {
143: $outstr.='normal:';
144: }
145: $outstr.='cond';
1.2 www 146: } elsif ($token->[1] eq 'link') {
1.3 www 147: # ----------------------------------------------------------------------- Links
1.2 www 148: $outstr.='<&>objlinks';
1.7 www 149:
1.3 www 150: if ($token->[2]->{'index'}) {
1.4 www 151: if ($links[$token->[2]->{'index'}]) {
152: $errtext.='Error: multiple use of link index '.
1.3 www 153: $token->[2]->{'index'}.'. ';
1.4 www 154: }
155: $outstr.='<:>'.$token->[2]->{'index'};
156: $links[$token->[2]->{'index'}]=1;
157: } else {
158: my $i=1;
159: while (($i<=$#links) && ($links[$i]==1)) { $i++; }
160: $outstr.='<:>'.$i;
161: $links[$i]=1;
162: }
1.7 www 163:
1.2 www 164: $outstr.='<:>'.$token->[2]->{'from'}.
1.5 www 165: ':'.$token->[2]->{'to'};
1.2 www 166: if ($token->[2]->{'condition'}) {
1.5 www 167: $outstr.=':'.$token->[2]->{'condition'};
1.2 www 168: } else {
1.5 www 169: $outstr.=':0';
1.4 www 170: }
1.11 www 171: # ------------------------------------------------------------------- Parameter
172: } elsif ($token->[1] eq 'param') {
173: $outstr.='<&>objparms<:>'.$token->[2]->{'to'}.'<:>'.
1.13 www 174: $token->[2]->{'type'}.'___'.$token->[2]->{'name'}
1.11 www 175: .'___'.$token->[2]->{'value'};
1.2 www 176: } elsif ($graphmode) {
1.3 www 177: # --------------------------------------------- All other tags (graphical only)
178: $outstr.='<&>'.$token->[1];
1.4 www 179: if (defined($token->[2]->{'index'})) {
1.3 www 180: $outstr.='<:>'.$token->[2]->{'index'};
181: if ($token->[1] eq 'obj') {
182: $obj[$token->[2]->{'index'}]=2;
183: }
184: }
185: $outstr.='<:>'.$token->[2]->{'value'};
1.2 www 186: }
187: }
188: }
189:
190: } else {
1.3 www 191: $errtext.='Map not loaded: The file does not exist. ';
1.2 www 192: }
1.28 www 193: return($outstr,$errtext,$infotext);
1.2 www 194: }
195:
196:
197: # ----------------------------------------------------------- Saves map to disk
198:
199: sub savemap {
1.20 albertel 200: my ($fn,$errtext)=@_;
1.28 www 201: my $infotext='';
1.13 www 202: my %alltypes;
203: my %allvalues;
1.22 www 204: if (($fn=~/\.sequence(\.tmp)*$/) ||
205: ($fn=~/\.page(\.tmp)*$/)) {
1.4 www 206:
1.2 www 207: # ------------------------------------------------------------- Deal with input
1.30 ! albertel 208: my @tags=split(/<&>/,$env{'form.output'});
1.2 www 209: my $outstr='';
210: my $graphdef=0;
211: if ($tags[0] eq 'graphdef<:>yes') {
212: $outstr='<map mode="rat/graphical">'."\n";
213: $graphdef=1;
214: } else {
215: $outstr="<map>\n";
216: }
1.23 www 217: foreach (@tags) {
1.2 www 218: my @parts=split(/<:>/,$_);
219: if ($parts[0] eq 'objcont') {
220: my @comp=split(/:/,$parts[$#parts]);
221: # --------------------------------------------------------------- Logical input
222: if ($comp[$#comp] eq 'res') {
1.4 www 223: $comp[0]=qtescape($comp[0]);
224: $comp[1]=qtescape($comp[1]);
1.2 www 225: if ($comp[2] eq 'true') {
226: if ($comp[1]!~/^http\:\/\//) {
227: $comp[1]='http://'.$comp[1];
228: }
1.14 www 229: $comp[1].='" external="true';
1.2 www 230: } else {
231: if ($comp[1]=~/^http\:\/\//) {
232: $comp[1]=~s/^http\:\/\/[^\/]*\//\//;
233: }
234: }
235: $outstr.='<resource id="'.$parts[1].'" src="'
1.4 www 236: .$comp[1].'"';
1.2 www 237:
238: if (($comp[3] ne '') && ($comp[3] ne 'normal')) {
239: $outstr.=' type="'.$comp[3].'"';
240: }
241: if ($comp[0] ne '') {
1.4 www 242: $outstr.=' title="'.$comp[0].'"';
1.2 www 243: }
244: $outstr.="></resource>\n";
245: } elsif ($comp[$#comp] eq 'cond') {
246: $outstr.='<condition id="'.$parts[1].'"';
247: if (($comp[1] ne '') && ($comp[1] ne 'normal')) {
248: $outstr.=' type="'.$comp[1].'"';
249: }
250: $outstr.=' value="'.qtescape($comp[0]).'"';
251: $outstr.="></condition>\n";
252: }
253: } elsif ($parts[0] eq 'objlinks') {
254: my @comp=split(/:/,$parts[$#parts]);
255: $outstr.='<link';
256: $outstr.=' from="'.$comp[0].'"';
257: $outstr.=' to="'.$comp[1].'"';
258: if (($comp[2] ne '') && ($comp[2]!=0)) {
259: $outstr.=' condition="'.$comp[2].'"';
260: }
261: $outstr.=' index="'.$parts[1].'"';
262: $outstr.="></link>\n";
1.11 www 263: } elsif ($parts[0] eq 'objparms') {
1.13 www 264: undef %alltypes;
265: undef %allvalues;
1.20 albertel 266: foreach (split(/:/,$parts[$#parts])) {
1.11 www 267: my ($type,$name,$value)=split(/\_\_\_/,$_);
1.13 www 268: $alltypes{$name}=$type;
269: $allvalues{$name}=$value;
1.20 albertel 270: }
271: foreach (keys %allvalues) {
272: if ($allvalues{$_} ne '') {
1.13 www 273: $outstr.='<param to="'.$parts[1].'" type="'
274: .$alltypes{$_}.'" name="'.$_
275: .'" value="'.$allvalues{$_}.'">'
1.12 www 276: ."</param>\n";
1.20 albertel 277: }
278: }
1.2 www 279: } elsif (($parts[0] ne '') && ($graphdef)) {
280: # ------------------------------------------------------------- Graphical input
281: $outstr.='<'.$parts[0];
282: if ($#parts==2) {
283: $outstr.=' index="'.$parts[1].'"';
284: }
285: $outstr.=' value="'.qtescape($parts[$#parts]).'"></'.
286: $parts[0].">\n";
287: }
1.23 www 288: }
1.2 www 289: $outstr.="</map>\n";
1.26 raeburn 290: if ($fn=~/^\/*uploaded\/(\w+)\/(\w+)\/(.*)$/) {
1.30 ! albertel 291: $env{'form.output'}=$outstr;
1.23 www 292: my $home=&Apache::lonnet::homeserver($2,$1);
1.25 albertel 293: my $result=&Apache::lonnet::finishuserfileupload($2,$1,$home,
1.26 raeburn 294: 'output',$3);
1.25 albertel 295: if ($result != m|^/uploaded/|) {
296: $errtext.='Map not saved: A network error occured when trying to save the map. ';
297: }
1.21 www 298: } else {
1.2 www 299: my $fh;
300: if ($fh=Apache::File->new(">$fn")) {
301: print $fh $outstr;
1.28 www 302: $infotext.="Map saved as $fn. ";
1.2 www 303: } else {
1.17 matthew 304: $errtext.='Could not write file '.$fn.'. Map not saved. ';
1.2 www 305: }
306: }
307: } else {
308: # -------------------------------------------- Cannot write to that file, error
1.20 albertel 309: $errtext.='Map not saved: The specified path does not exist. ';
1.2 www 310: }
1.28 www 311: return ($errtext,$infotext);
1.2 www 312: }
1.1 www 313:
314: # ================================================================ Main Handler
315:
316: sub handler {
317: my $r=shift;
1.29 albertel 318: &Apache::loncommon::content_type($r,'text/html');
1.1 www 319: $r->send_http_header;
320:
321: return OK if $r->header_only;
322:
323: my $url=$r->uri;
1.2 www 324: $url=~/\/(\w+)\/ratserver$/;
325: my $mode=$1;
326:
327: $url=~s/\/loadonly\/ratserver$/\/save\/ratserver/;
328:
329: my $fn=$r->filename;
1.19 albertel 330: my $lonDocRoot=$r->dir_config('lonDocRoot');
331: if ( $fn =~ /$lonDocRoot/ ) {
332: #internal authentication, needs fixup.
333: $fn = $url;
334: $fn=~s|^/~(\w+)|/home/$1/public_html|;
335: $fn=~s|/[^/]*/ratserver$||;
336: }
1.2 www 337: my $errtext='';
1.28 www 338: my $infotext='';
1.2 www 339: my $outtext='';
340:
341: if ($mode ne 'loadonly') {
1.28 www 342: ($errtext,$infotext)=&savemap($fn,$errtext);
1.2 www 343: }
1.28 www 344: ($outtext,$errtext,$infotext)=&loadmap($fn,$errtext,$infotext);
1.1 www 345:
346: $r->print(<<ENDDOCUMENT);
347: <html>
1.8 harris41 348: <body bgcolor="#FFFFFF">
1.2 www 349: <form name=storage method=post action="$url">
350: <input type=hidden name=output value="$outtext">
1.1 www 351: </form>
1.8 harris41 352: <script>
1.9 harris41 353: parent.flag=1;
1.8 harris41 354: </script>
1.2 www 355: ENDDOCUMENT
1.28 www 356: if (($errtext ne '') || ($infotext ne '')) {
1.2 www 357: $r->print(<<ENDSCRIPT);
358: <script>
1.28 www 359: alert("$infotext $errtext");
1.2 www 360: </script>
361: ENDSCRIPT
362: }
363: $r->print("</body>\n</html>\n");
1.1 www 364:
365: return OK;
366: }
367:
368: 1;
369: __END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>