]> git.llucax.com Git - software/sercom-old.git/blob - src/sc_suwi
Bugfix (variable que se inicializaba condicionalmente).
[software/sercom-old.git] / src / sc_suwi
1 #!/usr/bin/env python2.4
2 # -*- encoding: iso-8859-1 -*-
3 # vim: set et sw=4 sts=4 :
4
5 # Módulos estándar
6 import os
7 import sys
8 import cgi
9 # Módulos externos
10 import sqlobject
11 # Módulos locales
12 import sercom
13 from sercom.sqlo import *
14
15 # Inicializo
16 conf, conn, log = sercom.init('cgi')
17
18 # Para debug web
19 import cgitb; cgitb.enable()
20
21 #XXX HORRIBLE
22 PASSWD = conf.get('general', 'cgipw')
23
24 def http_header_html(req):
25     return 'Content-type: text/html\r\n\r\n'
26
27 def http_header_zip(req, filename):
28     return 'Content-type: application/zip\r\n' \
29         'Content-Disposition: attachment;filename=%s\r\n' \
30         '\r\n' % filename
31
32 def header(req):
33     return '''<!DOCTYPE html
34     PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN"
35     "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd" >
36 <html>
37     <head>
38         <title>SUWI - Sercom Ugly Web Interface</title>
39         <style type="text/css">
40             <!--
41             body
42             {
43                 font-family: sans-serif;
44                 background-color: rgb(255, 255, 255);
45             }
46             table
47             {
48                 border: medium black solid;
49                 border-collapse: collapse;
50             }
51             th
52             {
53                 border: thin black solid;
54                 color: white;
55                 background-color: navy;
56                 padding: 3pt;
57                 text-align: center;
58                 vertical-align: middle;
59             }
60             td
61             {
62                 border: thin black solid;
63                 padding: 3pt;
64                 text-align: center;
65                 vertical-align: middle;
66             }
67             .warn
68             {
69                 color: red;
70             }
71             // -->
72         </style>
73     </head>
74     <body>
75 '''
76     pass
77
78 def footer(req):
79     return '''
80     </body>
81 </html>
82 '''
83     pass
84
85 def form(req, submit, str=None):
86     r = submit
87     if submit is not None:
88         r = '<form method="post">\n'
89         if str: r += str
90         r += '<input type="submit" value="%s">\n' % submit
91         r += '</form>\n'
92     return r
93
94 def pre(s):
95     return '<pre>%s</pre>' % cgi.escape(str(s))
96
97 def input_login(req):
98     return '<input type="hidden" name="pw" value="%s" />\n' % PASSWD
99
100 def input_curso(req, curso_id):
101     return input_login(req) \
102         + '<input type="hidden" name="curso" value="%d" />\n' % curso_id
103
104 def input_entrega(req, entrega_id):
105     e = Entrega.get(entrega_id, connection=conn)
106     return input_curso(req, e.curso.id) \
107         + '<input type="hidden" name="entrega" value="%d" />\n' % entrega_id
108
109 def input_zip(req, entrega_id):
110     return input_entrega(req, entrega_id) \
111         + '<input type="hidden" name="zip" value="1" />\n'
112
113 def input_inscripto(req, inscripto_id=None):
114     if inscripto_id is not None:
115         return '<input type="hidden" name="inscripto" value="%d" />\n' \
116             % inscripto_id
117     return ''
118
119 def input_pruebas(req, intento_id, inscripto_id=None):
120     i = Intento.get(intento_id, connection=conn)
121     r = input_entrega(req, i.entrega.id)
122     r += '<input type="hidden" name="intento" value="%d" />\n' % intento_id
123     r += input_inscripto(req, inscripto_id)
124     return r
125
126 def input_intentos(req, entrega_id, inscripto_id):
127     return input_entrega(req, entrega_id) + input_inscripto(req, inscripto_id)
128
129 def login(req):
130     r = '<h1>Bienvenido a ' \
131         '<acronym title="Sercom Ugly Web Interface">SUWI</acronym></h1>\n'
132     r += '<p>Debe ingresar la contraseña para seguir...</p>\n'
133     return r + form(req, 'Entrar',
134         '<p>Contraseña: <input type="password" name="pw" /></p>')
135
136 def curso(req):
137     cursos = [c for c in Curso.select(connection=conn) if len(c.entregas)]
138     r = '<h1>Elegir curso</h1>\n'
139     if cursos:
140         r += '<p>Curso: <select name="curso">\n'
141         for c in cursos:
142             r += '\t<option value="%d">%d-%d-%d</option>\n' \
143                 % (c.id, c.anio, c.cuatrimestre, c.curso)
144         r += '</select></p>\n'
145     else:
146         r += '<p>No hay cursos con entregas</p>\n'
147     return form(req, 'Ver', r + input_login(req))
148
149 def entrega(req, curso_id):
150     c = Curso.get(curso_id, connection=conn)
151     r = '<h1>Elegir entrega del curso %d-%d-%d</h1>\n' \
152         % (c.anio, c.cuatrimestre, c.curso)
153     r += '<p>Entrega: <select name="entrega">\n'
154     for e in c.entregas:
155         r += '\t<option value="%d">%d-%d</option>\n' \
156             % (e.id, e.nroEjercicio, e.entrega)
157     r += '</select></p>\n'
158     return form(req, 'Ver', r + input_curso(req, curso_id))
159
160 def pruebas(req, intento_id, inscripto_id=None):
161     def header():
162         return '''<table>
163     <tr>
164         <th>Nombre</th>
165         <th>Pasada</th>
166         <th>Privada</th>
167         <th>Activa</th>
168         <th>Inicio</th>
169         <th>Fin</th>
170         <th>Observaciones</th>
171     <tr>
172 '''
173         pass
174     def footer():
175         r = '</table>\n<p>\n'
176         r += form(req, 'Volver', input_entrega(req, i.entrega.id)
177             + input_inscripto(req, inscripto_id))
178         r += '</p>\n'
179         return r
180     def row(p):
181         return '''
182     <tr>
183         <td>%s</td>
184         <td>%s</td>
185         <td>%s</td>
186         <td>%s</td>
187         <td>%s</td>
188         <td>%s</td>
189         <td>%s</td>
190     </tr>
191 ''' % (p.casoDePrueba.nombre, p.pasada, p.casoDePrueba.privado,
192         p.casoDePrueba.activo, p.inicio, p.fin,
193         pre(p.observaciones))
194
195     i = Intento.get(intento_id, connection=conn)
196     r = '<h1>Pruebas del intento %d del alumno %d</h1>\n' \
197         % (i.numero, i.inscripto.padron)
198     r += header()
199     for p in i.pruebas:
200         r += row(p)
201     r += footer()
202     return r
203
204 def intentos(req, entrega_id, inscripto_id):
205     def header():
206         return '''<table>
207     <tr>
208         <th>Nro</th>
209         <th>Llegada</th>
210         <th>Compila</th>
211         <th>Pruebas</th>
212         <th>Observaciones</th>
213     <tr>
214 '''
215         pass
216     def footer():
217         r = '</table>\n<p>\n'
218         r += form(req, 'Volver', input_entrega(req, entrega_id))
219         r += '</p>\n'
220         return r
221     def row(i):
222         if i.pruebasPasadas: pruebas = 'BIEN'
223         elif not i.compila: pruebas = None
224         else: pruebas = 'MAL'
225         return '''
226     <tr>
227         <td>%d</td>
228         <td>%s</td>
229         <td>%s</td>
230         <td>%s</td>
231         <td>%s</td>
232     </tr>
233 ''' % (i.numero, i.llegada, i.compila,
234         form(req, pruebas, input_pruebas(req, i.id, inscripto_id)),
235         pre(i.observaciones))
236
237     r = '<h1>Intentos del alumno %d</h1>\n' \
238         % Inscripto.get(inscripto_id, connection=conn).padron
239     r += header()
240     for i in Intento.selectBy(entregaID=entrega_id, inscriptoID=inscripto_id,
241             connection=conn):
242         r += row(i)
243     r += footer()
244     return r
245
246 def zip(req, entrega_id):
247     def zip_path(path, base, zipfd):
248         paths = os.listdir(path)
249         for p in paths:
250             if os.path.isdir(p):
251                 zip_path(os.path.join(path, p), os.path.join(base, p), zipfd)
252             else:
253                 zipfd.write(os.path.join(path, p), os.path.join(base, p))
254
255     from zipfile import ZipFile, ZIP_DEFLATED
256     e = Entrega.get(entrega_id, connection=conn)
257     c = e.curso
258     req.write(http_header_zip(req, 'entrega-%d.%d.%d-%d.%d.zip'
259         % (c.anio, c.cuatrimestre, c.curso, e.nroEjercicio, e.entrega)))
260     tmpfname = os.tmpnam()
261     zipfd = ZipFile(tmpfname, 'w', ZIP_DEFLATED)
262     for c in e.correcciones:
263         zip_path(os.path.join(conf.get('general', 'data_dir'), c.intento.path),
264             str(c.intento.inscripto.padron), zipfd)
265     zipfd.close()
266     req.write(file(tmpfname, 'r').read())
267     os.unlink(tmpfname)
268
269 def correcciones(req, entrega_id):
270     def header():
271         return '''<table>
272     <tr>
273         <th>Padrón</th>
274         <th>Pruebas</th>
275         <th>Intentos</th>
276     <tr>
277 '''
278         pass
279     def footer():
280         r = '</table>\n<p>\n'
281         r += form(req, 'Elegir curso', input_login(req))
282         r += form(req, 'Elegir entrega', input_curso(req, e.curso.id))
283         r += form(req, 'Bajar entrega en .zip', input_zip(req, entrega_id))
284         r += '</p>\n'
285         return r
286     def row(c):
287         if c.intento.pruebasPasadas: pruebas = 'BIEN'
288         elif not c.intento.compila: prubas = None
289         else: pruebas = 'MAL'
290         intentos = int(Intento.selectBy(inscriptoID=c.inscriptoID,
291             entregaID=c.entregaID, connection=conn).count())
292         return '''
293     <tr>
294         <td>%d</td>
295         <td>%s</td>
296         <td>%s</td>
297     </tr>
298 ''' % (c.inscripto.padron,
299         form(req, pruebas, input_pruebas(req, c.intento.id)),
300         form(req, intentos, input_intentos(req, c.entrega.id, c.inscripto.id)))
301
302     e = Entrega.get(entrega_id, connection=conn)
303     c = e.curso
304     r = '<h1>Entrega %d.%d del curso %d-%d-%d</h1>\n' \
305         % (e.nroEjercicio, e.entrega, c.anio, c.cuatrimestre, c.curso)
306     r += header()
307     for c in e.correcciones:
308         r += row(c)
309     r += footer()
310     return r
311
312 #
313 # MAIN
314 #
315
316 req = sys.stdout
317 f = cgi.FieldStorage()
318 raw = f.has_key('zip')
319
320
321 if not raw:
322     print http_header_html(req)
323     print header(req)
324
325 path = os.getenv('PATH_INFO')
326
327 if not f.has_key('pw'):
328     print login(req)
329 elif f.has_key('pw') and f.getfirst('pw') <> PASSWD:
330     print login(req)
331     print '<p class="warn">Password Incorrecto!</p>\n'
332 elif not f.has_key('curso'):
333     print curso(req)
334 elif not f.has_key('entrega'):
335     print entrega(req, int(f.getfirst('curso')))
336 elif f.has_key('intento'):
337     inscripto_id = None
338     if f.has_key('inscripto'):
339         inscripto_id = int(f.getfirst('inscripto'))
340     print pruebas(req, int(f.getfirst('intento')), inscripto_id)
341 elif f.has_key('inscripto'):
342     print intentos(req, int(f.getfirst('entrega')), int(f.getfirst('inscripto')))
343 elif f.has_key('zip'):
344     zip(req, int(f.getfirst('entrega')))
345 else:
346     print correcciones(req, int(f.getfirst('entrega')))
347
348 if not raw:
349     print footer(req)
350