Commit 9caa0ee3836526b7006836d22f1b96fda3bd4874
1 parent
1245246a
Exists in
master
and in
1 other branch
Added decorator @admin_only to the handlers that are restricted to the administrator.
This also uniformizes the http response to 403 Forbidden.
Showing
2 changed files
with
57 additions
and
47 deletions
 
Show diff stats
BUGS.md
| 1 | 1 | |
| 2 | 2 | # BUGS | 
| 3 | 3 | |
| 4 | +- self.testapp.get_json_filename_of_test(test_id) retorna None quando test_id nao existe. | |
| 4 | 5 | - a revisao do teste não mostra as imagens. | 
| 5 | 6 | - se aluno tem teste activo e é allowed uma segunda vez, deve manter o mesmo teste. adicionar opcao para eliminar um teste em curso. | 
| 6 | 7 | - melhorar o botao de autorizar (desliga-se), usar antes um botao? | ... | ... | 
serve.py
| ... | ... | @@ -11,6 +11,7 @@ import mimetypes | 
| 11 | 11 | import signal | 
| 12 | 12 | import asyncio | 
| 13 | 13 | import json | 
| 14 | +import functools | |
| 14 | 15 | |
| 15 | 16 | # user installed libraries | 
| 16 | 17 | import tornado.ioloop | 
| ... | ... | @@ -22,10 +23,21 @@ from tornado import template, gen | 
| 22 | 23 | from app import App, AppException | 
| 23 | 24 | from tools import load_yaml, md_to_html | 
| 24 | 25 | |
| 26 | +# ---------------------------------------------------------------------------- | |
| 27 | +# Decorator used to restrict access only to the administrator | |
| 28 | +# ---------------------------------------------------------------------------- | |
| 29 | +def admin_only(func): | |
| 30 | + @functools.wraps(func) | |
| 31 | + def wrapper(self, *args, **kwargs): | |
| 32 | + if self.current_user != '0': | |
| 33 | + raise tornado.web.HTTPError(403) # forbidden | |
| 34 | + else: | |
| 35 | + func(self, *args, **kwargs) | |
| 36 | + return wrapper | |
| 25 | 37 | |
| 26 | -# ------------------------------------------------------------------------- | |
| 38 | +# ---------------------------------------------------------------------------- | |
| 27 | 39 | # Web Application. Routes to handler classes. | 
| 28 | -# ------------------------------------------------------------------------- | |
| 40 | +# ---------------------------------------------------------------------------- | |
| 29 | 41 | class WebApplication(tornado.web.Application): | 
| 30 | 42 | def __init__(self, testapp, debug=False): | 
| 31 | 43 | handlers = [ | 
| ... | ... | @@ -52,9 +64,9 @@ class WebApplication(tornado.web.Application): | 
| 52 | 64 | self.testapp = testapp | 
| 53 | 65 | |
| 54 | 66 | |
| 55 | -# ------------------------------------------------------------------------- | |
| 67 | +# ---------------------------------------------------------------------------- | |
| 56 | 68 | # Base handler. Other handlers will inherit this one. | 
| 57 | -# ------------------------------------------------------------------------- | |
| 69 | +# ---------------------------------------------------------------------------- | |
| 58 | 70 | class BaseHandler(tornado.web.RequestHandler): | 
| 59 | 71 | @property | 
| 60 | 72 | def testapp(self): | 
| ... | ... | @@ -66,9 +78,9 @@ class BaseHandler(tornado.web.RequestHandler): | 
| 66 | 78 | return cookie.decode('utf-8') | 
| 67 | 79 | |
| 68 | 80 | |
| 69 | -# ------------------------------------------------------------------------- | |
| 81 | +# ---------------------------------------------------------------------------- | |
| 70 | 82 | # /login | 
| 71 | -# ------------------------------------------------------------------------- | |
| 83 | +# ---------------------------------------------------------------------------- | |
| 72 | 84 | class LoginHandler(BaseHandler): | 
| 73 | 85 | SUPPORTED_METHODS = ['GET', 'POST'] | 
| 74 | 86 | |
| ... | ... | @@ -84,12 +96,12 @@ class LoginHandler(BaseHandler): | 
| 84 | 96 | self.set_secure_cookie("user", str(uid), expires_days=30) | 
| 85 | 97 | self.redirect(self.get_argument("next", "/")) | 
| 86 | 98 | else: | 
| 87 | - self.render("login.html", | |
| 88 | - error='Não autorizado ou número/senha inválido') | |
| 99 | + self.render("login.html", error='Não autorizado ou número/senha inválido') | |
| 89 | 100 | |
| 90 | -# ------------------------------------------------------------------------- | |
| 101 | + | |
| 102 | +# ---------------------------------------------------------------------------- | |
| 91 | 103 | # /logout | 
| 92 | -# ------------------------------------------------------------------------- | |
| 104 | +# ---------------------------------------------------------------------------- | |
| 93 | 105 | class LogoutHandler(BaseHandler): | 
| 94 | 106 | @tornado.web.authenticated | 
| 95 | 107 | def get(self): | 
| ... | ... | @@ -101,10 +113,24 @@ class LogoutHandler(BaseHandler): | 
| 101 | 113 | |
| 102 | 114 | |
| 103 | 115 | # ---------------------------------------------------------------------------- | 
| 116 | +# handles root / to redirect students to /test and admininistrator to /admin | |
| 117 | +# ---------------------------------------------------------------------------- | |
| 118 | +# FIXME list available tests | |
| 119 | +class RootHandler(BaseHandler): | |
| 120 | + @tornado.web.authenticated | |
| 121 | + def get(self): | |
| 122 | + if self.current_user == '0': | |
| 123 | + self.redirect('/admin') | |
| 124 | + else: | |
| 125 | + self.redirect('/test') | |
| 126 | + | |
| 127 | + | |
| 128 | +# ---------------------------------------------------------------------------- | |
| 104 | 129 | # Serves files from the /public subdir of the topics. | 
| 105 | 130 | # Based on https://bhch.github.io/posts/2017/12/serving-large-files-with-tornado-safely-without-blocking/ | 
| 106 | 131 | # ---------------------------------------------------------------------------- | 
| 107 | 132 | class FileHandler(BaseHandler): | 
| 133 | + SUPPORTED_METHODS = ['GET'] | |
| 108 | 134 | chunk_size = 512 * 1024 # serve up to 512 KiB multiple times | 
| 109 | 135 | |
| 110 | 136 | @tornado.web.authenticated | 
| ... | ... | @@ -171,14 +197,15 @@ class TestHandler(BaseHandler): | 
| 171 | 197 | 'success': 'question-success.html', | 
| 172 | 198 | } | 
| 173 | 199 | |
| 174 | - # GET | |
| 200 | + # --- GET | |
| 175 | 201 | @tornado.web.authenticated | 
| 176 | 202 | def get(self): | 
| 177 | 203 | uid = self.current_user | 
| 204 | + # FIXME make generate async? | |
| 178 | 205 | t = self.testapp.get_student_test(uid) or self.testapp.generate_test(uid) | 
| 179 | 206 | self.render('test.html', t=t, md=md_to_html, templ=self._templates) | 
| 180 | 207 | |
| 181 | - # POST | |
| 208 | + # --- POST | |
| 182 | 209 | @tornado.web.authenticated | 
| 183 | 210 | async def post(self): | 
| 184 | 211 | uid = self.current_user | 
| ... | ... | @@ -211,6 +238,19 @@ class TestHandler(BaseHandler): | 
| 211 | 238 | self.render('grade.html', t=t, allgrades=self.testapp.get_student_grades_from_all_tests(uid)) | 
| 212 | 239 | |
| 213 | 240 | |
| 241 | +# ------------------------------------------------------------------------- | |
| 242 | +# FIXME this should be a post in the test with command giveup instead of correct... | |
| 243 | +# class GiveupHandler(BaseHandler): | |
| 244 | +# @tornado.web.authenticated | |
| 245 | +# def get(self): | |
| 246 | +# uid = self.current_user | |
| 247 | +# t = self.testapp.giveup_test(uid) | |
| 248 | +# self.testapp.logout(uid) | |
| 249 | + | |
| 250 | +# # --- Show result to student | |
| 251 | +# self.render('grade.html', t=t, allgrades=self.testapp.get_student_grades_from_all_tests(uid)) | |
| 252 | + | |
| 253 | + | |
| 214 | 254 | # --- REVIEW ------------------------------------------------------------- | 
| 215 | 255 | class ReviewHandler(BaseHandler): | 
| 216 | 256 | _templates = { | 
| ... | ... | @@ -230,15 +270,12 @@ class ReviewHandler(BaseHandler): | 
| 230 | 270 | } | 
| 231 | 271 | |
| 232 | 272 | @tornado.web.authenticated | 
| 273 | + @admin_only | |
| 233 | 274 | def get(self): | 
| 234 | - uid = self.current_user | |
| 235 | - if uid != '0': | |
| 236 | - raise tornado.web.HTTPError(404) | |
| 237 | - | |
| 238 | 275 | test_id = self.get_query_argument('test_id', None) | 
| 239 | 276 | logging.info(f'Review test {test_id}.') | 
| 240 | 277 | try: | 
| 241 | - fname = self.testapp.get_json_filename_of_test(test_id) | |
| 278 | + fname = self.testapp.get_json_filename_of_test(test_id) # FIXME is returning None if nonexistent | |
| 242 | 279 | except: | 
| 243 | 280 | raise tornado.web.HTTPError(404, 'Test ID not found.') | 
| 244 | 281 | |
| ... | ... | @@ -253,38 +290,12 @@ class ReviewHandler(BaseHandler): | 
| 253 | 290 | |
| 254 | 291 | |
| 255 | 292 | # ------------------------------------------------------------------------- | 
| 256 | -# FIXME this should be a post in the test with command giveup instead of correct... | |
| 257 | -class GiveupHandler(BaseHandler): | |
| 258 | - @tornado.web.authenticated | |
| 259 | - def get(self): | |
| 260 | - uid = self.current_user | |
| 261 | - t = self.testapp.giveup_test(uid) | |
| 262 | - self.testapp.logout(uid) | |
| 263 | - | |
| 264 | - # --- Show result to student | |
| 265 | - self.render('grade.html', t=t, allgrades=self.testapp.get_student_grades_from_all_tests(uid)) | |
| 266 | - | |
| 267 | - | |
| 268 | -# ------------------------------------------------------------------------- | |
| 269 | -# FIXME list available tests | |
| 270 | -class RootHandler(BaseHandler): | |
| 271 | - @tornado.web.authenticated | |
| 272 | - def get(self): | |
| 273 | - if self.current_user == '0': | |
| 274 | - self.redirect('/admin') | |
| 275 | - else: | |
| 276 | - self.redirect('/test') | |
| 277 | - | |
| 278 | - | |
| 279 | -# ------------------------------------------------------------------------- | |
| 280 | 293 | class AdminHandler(BaseHandler): | 
| 281 | 294 | SUPPORTED_METHODS = ['GET', 'POST'] | 
| 282 | 295 | |
| 283 | 296 | @tornado.web.authenticated | 
| 297 | + @admin_only | |
| 284 | 298 | def get(self): | 
| 285 | - if self.current_user != '0': | |
| 286 | - raise tornado.web.HTTPError(403) | |
| 287 | - | |
| 288 | 299 | cmd = self.get_query_argument('cmd', default=None) | 
| 289 | 300 | |
| 290 | 301 | if cmd == 'students_table': | 
| ... | ... | @@ -307,10 +318,8 @@ class AdminHandler(BaseHandler): | 
| 307 | 318 | self.render('admin.html') | 
| 308 | 319 | |
| 309 | 320 | @tornado.web.authenticated | 
| 321 | + @admin_only | |
| 310 | 322 | def post(self): | 
| 311 | - if self.current_user != '0': | |
| 312 | - self.redirect('/') | |
| 313 | - | |
| 314 | 323 | cmd = self.get_body_argument('cmd', None) | 
| 315 | 324 | value = self.get_body_argument('value', None) | 
| 316 | 325 | ... | ... |