Commit a08905e35197e2fa9b5e9c0b2187c6d3754cce63
1 parent
a305bb5d
Exists in
master
and in
1 other branch
- improved security against cross-site request forgery (XRSF).
Showing
4 changed files
with
30 additions
and
32 deletions
Show diff stats
serve.py
... | ... | @@ -39,7 +39,7 @@ class WebApplication(tornado.web.Application): |
39 | 39 | 'template_path': path.join(path.dirname(__file__), 'templates'), |
40 | 40 | 'static_path': path.join(path.dirname(__file__), 'static'), |
41 | 41 | 'static_url_prefix': '/static/', |
42 | - 'xsrf_cookies': False, # FIXME not needed on private network | |
42 | + 'xsrf_cookies': True, | |
43 | 43 | 'cookie_secret': base64.b64encode(uuid.uuid4().bytes), |
44 | 44 | 'login_url': '/login', |
45 | 45 | 'debug': debug, |
... | ... | @@ -105,7 +105,7 @@ class LogoutHandler(BaseHandler): |
105 | 105 | # Based on https://bhch.github.io/posts/2017/12/serving-large-files-with-tornado-safely-without-blocking/ |
106 | 106 | # ---------------------------------------------------------------------------- |
107 | 107 | class FileHandler(BaseHandler): |
108 | - chunk_size = 1024 * 1024 # serve up to 1 MiB multiple times | |
108 | + chunk_size = 512 * 1024 # serve up to 512 KiB multiple times | |
109 | 109 | |
110 | 110 | @tornado.web.authenticated |
111 | 111 | async def get(self): | ... | ... |
static/js/admin.js
1 | +// from: https://www.tornadoweb.org/en/stable/guide/security.html | |
2 | +// with changes: removed datatype and callback from original postJSON | |
3 | +function getCookie(name) { | |
4 | + var r = document.cookie.match("\\b" + name + "=([^;]*)\\b"); | |
5 | + return r ? r[1] : undefined; | |
6 | +} | |
7 | + | |
8 | +jQuery.postJSON = function(url, args) { | |
9 | + args._xsrf = getCookie("_xsrf"); | |
10 | + $.ajax({url: url, data: $.param(args), type: "POST"}); | |
11 | +}; | |
12 | + | |
13 | +// --------------------------------------------------------------------------- | |
1 | 14 | $(document).ready(function() { |
2 | 15 | function button_handlers() { |
3 | 16 | // button handlers (runs once) |
... | ... | @@ -13,28 +26,20 @@ $(document).ready(function() { |
13 | 26 | ); |
14 | 27 | $("#reset_password").click( |
15 | 28 | function () { |
16 | - $.ajax({ | |
17 | - type: "POST", | |
18 | - url: "/admin", | |
19 | - data: { | |
20 | - "cmd": "reset_password", | |
21 | - "value": $("#reset_number").val() | |
22 | - } | |
29 | + $.postJSON("/admin", { | |
30 | + "cmd": "reset_password", | |
31 | + "value": $("#reset_number").val() | |
23 | 32 | }); |
24 | 33 | } |
25 | 34 | ); |
26 | 35 | $("#inserir_novo_aluno").click( |
27 | 36 | function () { |
28 | - $.ajax({ | |
29 | - type: "POST", | |
30 | - url: "/admin", | |
31 | - data: { | |
32 | - "cmd": "insert_student", | |
33 | - "value": JSON.stringify({ | |
34 | - "number": $("#novo_numero").val(), | |
35 | - "name": $("#novo_nome").val() | |
36 | - }) | |
37 | - } | |
37 | + $.postJSON("/admin", { | |
38 | + "cmd": "insert_student", | |
39 | + "value": JSON.stringify({ | |
40 | + "number": $("#novo_numero").val(), | |
41 | + "name": $("#novo_nome").val() | |
42 | + }) | |
38 | 43 | }); |
39 | 44 | } |
40 | 45 | ); |
... | ... | @@ -46,20 +51,12 @@ $(document).ready(function() { |
46 | 51 | // checkbox handler to allow/deny students individually |
47 | 52 | function autorizeStudent(e) { |
48 | 53 | if (this.checked) { |
49 | - $(this).parent().parent().addClass("table-primary"); // row class | |
50 | - $.ajax({ | |
51 | - type: "POST", | |
52 | - url: "/admin", | |
53 | - data: {"cmd": "allow", "value": this.name} | |
54 | - }); | |
54 | + // $(this).parent().parent().addClass("table-primary"); // row class | |
55 | + $.postJSON("/admin", {"cmd": "allow", "value": this.name}); | |
55 | 56 | } |
56 | 57 | else { |
57 | 58 | // $(this).parent().parent().removeClass("active"); |
58 | - $.ajax({ | |
59 | - type: "POST", | |
60 | - url: "/admin", | |
61 | - data: {"cmd": "deny", "value": this.name} | |
62 | - }); | |
59 | + $.postJSON("/admin", {"cmd": "deny", "value": this.name}); | |
63 | 60 | } |
64 | 61 | } |
65 | 62 | ... | ... |
templates/admin.html
... | ... | @@ -24,7 +24,7 @@ |
24 | 24 | <script defer src="/static/fontawesome.min.js"></script> |
25 | 25 | <script defer src="/static/popper.min.js"></script> |
26 | 26 | <script defer src="/static/bootstrap/js/bootstrap.min.js"></script> |
27 | - <script type="text/javascript" src="/static/DataTables/datatables.min.js"></script> | |
27 | + <script defer src="/static/DataTables/datatables.min.js"></script> | |
28 | 28 | |
29 | 29 | <script defer src="/static/js/admin.js"></script> |
30 | 30 | <script defer src="/static/js/clock.js"></script> |
... | ... | @@ -140,7 +140,6 @@ |
140 | 140 | <label for="reset_number" class="col-sm-2 col-form-label">Número</label> |
141 | 141 | <div class="col-sm-10"> |
142 | 142 | <input id="reset_number" type="text" class="form-control"> |
143 | - <!-- <input type="text" class="form-control" id="novo_numero" value=""> --> | |
144 | 143 | </div> |
145 | 144 | </div> |
146 | 145 | </div> | ... | ... |
templates/test.html
... | ... | @@ -92,6 +92,8 @@ |
92 | 92 | </div> |
93 | 93 | |
94 | 94 | <form action="/test" method="post" id="test" autocomplete="off"> |
95 | + {% module xsrf_form_html() %} | |
96 | + | |
95 | 97 | {% for i, q in enumerate(t['questions']) %} |
96 | 98 | {% module Template(templ[q['type']], i=i, q=q, md=md(q['ref']), show_ref=t['show_ref']) %} |
97 | 99 | {% end %} | ... | ... |