Compare commits
	
		
			19 Commits
		
	
	
		
			v0.5.2
			...
			95cea46a8f
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 95cea46a8f | |||
| 02a1129390 | |||
| 438e09f1ec | |||
| 9241e1c0f7 | |||
| 8deefb9035 | |||
| 4f2984deea | |||
| 70d2325579 | |||
| 36d840c752 | |||
| 4400446718 | |||
| adead30a77 | |||
| 487f24732d | |||
| 3c06cebddf | |||
| d1d52fa4b6 | |||
| 80dc8b3cff | |||
| a9ccd64de2 | |||
| f5b9758bb1 | |||
| 84570d5974 | |||
| edb8241ad3 | |||
| 644a539ed9 | 
| @@ -50,7 +50,7 @@ | |||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
|                             <h5 class="mb-1">Start Time</h5> |                             <h5 class="mb-1">Start Time</h5> | ||||||
|                         </div> |                         </div> | ||||||
|                         {{ entry.start_time.strftime('%d %b %Y %H:%M:%S') }} |                         {{ entry.start_time.strftime('%d %b %Y %H:%M:%S') if entry.start_time else None }} | ||||||
|                     </li> |                     </li> | ||||||
|                     <li class="list-group-item list-group-item-action"> |                     <li class="list-group-item list-group-item-action"> | ||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
| @@ -59,7 +59,7 @@ | |||||||
|                                 <span class="badge bg-danger">Late</span> |                                 <span class="badge bg-danger">Late</span> | ||||||
|                             {% endif %} |                             {% endif %} | ||||||
|                         </div> |                         </div> | ||||||
|                         {{ entry.end_time.strftime('%d %b %Y %H:%M:%S') }} |                         {{ entry.end_time.strftime('%d %b %Y %H:%M:%S') if entry.end_time else None }} | ||||||
|                     </li>        |                     </li>        | ||||||
|                     <li class="list-group-item list-group-item-action"> |                     <li class="list-group-item list-group-item-action"> | ||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
|   | |||||||
| @@ -1,2 +1,2 @@ | |||||||
| <p>This web app was developed by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | <p>This web app was developed and is maintained by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | ||||||
| <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | ||||||
|     <div class="container"> |     <div class="container"> | ||||||
|         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest (Beta) | Admin</a> |         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest | Admin</a> | ||||||
|         <button |         <button | ||||||
|             class="navbar-toggler" |             class="navbar-toggler" | ||||||
|             type="button" |             type="button" | ||||||
|   | |||||||
| @@ -28,7 +28,7 @@ | |||||||
|                                                     <a href="{{ url_for('admin._view_test', id=test.id) }}">{{ test.get_code() }}</a> |                                                     <a href="{{ url_for('admin._view_test', id=test.id) }}">{{ test.get_code() }}</a> | ||||||
|                                                 </td> |                                                 </td> | ||||||
|                                                 <td> |                                                 <td> | ||||||
|                                                     {{ test.end_date.strftime('%d %b %Y') }} |                                                     {{ test.end_date.strftime('%d %b %Y') if test.end_date else None }} | ||||||
|                                                 </td> |                                                 </td> | ||||||
|                                             </tr> |                                             </tr> | ||||||
|                                         {% endfor %} |                                         {% endfor %} | ||||||
| @@ -72,10 +72,14 @@ | |||||||
|                                                     <a href="{{ url_for('admin._view_entry', id=result.id) }}">{{ result.get_surname() }}, {{ result.get_first_name() }}</a> |                                                     <a href="{{ url_for('admin._view_entry', id=result.id) }}">{{ result.get_surname() }}, {{ result.get_first_name() }}</a> | ||||||
|                                                 </td> |                                                 </td> | ||||||
|                                                 <td> |                                                 <td> | ||||||
|                                                     {{ result.end_time.strftime('%d %b %Y %H:%M') }} |                                                     {{ result.end_time.strftime('%d %b %Y %H:%M') if result.end_time else None }} | ||||||
|                                                 </td> |                                                 </td> | ||||||
|                                                 <td> |                                                 <td> | ||||||
|                                                     {{ (100*result.result['score']/result.result['max'])|round|int }}% ({{ result.result.grade }}) |                                                     {% if result.result %} | ||||||
|  |                                                         {{ (100*result.result['score']/result.result['max'])|round|int }}% ({{ result.result.grade }}) | ||||||
|  |                                                     {% else %} | ||||||
|  |                                                         Incomplete | ||||||
|  |                                                     {% endif %} | ||||||
|                                                 </td> |                                                 </td> | ||||||
|                                             </tr> |                                             </tr> | ||||||
|                                         {% endfor %} |                                         {% endfor %} | ||||||
| @@ -117,7 +121,7 @@ | |||||||
|                                                         <a href="{{ url_for('admin._view_test', id=test.id) }}">{{ test.get_code() }}</a> |                                                         <a href="{{ url_for('admin._view_test', id=test.id) }}">{{ test.get_code() }}</a> | ||||||
|                                                     </td> |                                                     </td> | ||||||
|                                                     <td> |                                                     <td> | ||||||
|                                                         {{ test.end_date.strftime('%d %b %Y') }} |                                                         {{ test.end_date.strftime('%d %b %Y') if test.end_date else None }} | ||||||
|                                                     </td> |                                                     </td> | ||||||
|                                                 </tr> |                                                 </tr> | ||||||
|                                             {% endfor %} |                                             {% endfor %} | ||||||
| @@ -138,7 +142,7 @@ | |||||||
|                 <div class="card m-3"> |                 <div class="card m-3"> | ||||||
|                     <div class="card-body"> |                     <div class="card-body"> | ||||||
|                         <h5 class="card-title">Help</h5> |                         <h5 class="card-title">Help</h5> | ||||||
|                         <p class="card-text">This web app was developed by Vivek Santayana. If there are any issues with the app, any bugs you need to report, or any features you would like to request, please feel free to <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test/issues">open an issue at the Git Repository</a>.</p> |                         <p class="card-text">This web app was developed and is maintained by Vivek Santayana. If there are any issues with the app, any bugs you need to report, or any features you would like to request, please feel free to <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test/issues">open an issue at the Git Repository</a>.</p> | ||||||
|                         <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test/issues" class="btn btn-primary">Open an Issue</a> |                         <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test/issues" class="btn btn-primary">Open an Issue</a> | ||||||
|                     </div> |                     </div> | ||||||
|                   </div> |                   </div> | ||||||
|   | |||||||
| @@ -49,7 +49,7 @@ | |||||||
|                             <div class="d-flex w-100 justify-content-between"> |                             <div class="d-flex w-100 justify-content-between"> | ||||||
|                                 <h5 class="mb-1">Start Time</h5> |                                 <h5 class="mb-1">Start Time</h5> | ||||||
|                             </div> |                             </div> | ||||||
|                             {{ entry.start_time.strftime('%d %b %Y %H:%M:%S') }} |                             {{ entry.start_time.strftime('%d %b %Y %H:%M:%S') if entry.start_time else None }} | ||||||
|                         </li> |                         </li> | ||||||
|                     {% endif %} |                     {% endif %} | ||||||
|                     <li class="list-group-item list-group-item-action"> |                     <li class="list-group-item list-group-item-action"> | ||||||
|   | |||||||
| @@ -32,13 +32,13 @@ | |||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
|                             <h5 class="mb-1">Start Date</h5> |                             <h5 class="mb-1">Start Date</h5> | ||||||
|                         </div> |                         </div> | ||||||
|                         {{ test.start_date.strftime('%d %b %Y %H:%M') }} |                         {{ test.start_date.strftime('%d %b %Y %H:%M') if test.start_date else None }} | ||||||
|                     </li> |                     </li> | ||||||
|                     <li class="list-group-item list-group-item-action"> |                     <li class="list-group-item list-group-item-action"> | ||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
|                             <h5 class="mb-1">Expiry Date</h5> |                             <h5 class="mb-1">Expiry Date</h5> | ||||||
|                         </div> |                         </div> | ||||||
|                         {{ test.end_date.strftime('%d %b %Y %H:%M') }} |                         {{ test.end_date.strftime('%d %b %Y %H:%M') if test.end_date else None }} | ||||||
|                     </li> |                     </li> | ||||||
|                     <li class="list-group-item list-group-item-action"> |                     <li class="list-group-item list-group-item-action"> | ||||||
|                         <div class="d-flex w-100 justify-content-between"> |                         <div class="d-flex w-100 justify-content-between"> | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ from flask import abort, Blueprint, jsonify, render_template, request, send_file | |||||||
| from flask.helpers import abort, flash, redirect, url_for | from flask.helpers import abort, flash, redirect, url_for | ||||||
| from flask_login import current_user, login_required | from flask_login import current_user, login_required | ||||||
|  |  | ||||||
| from datetime import date, datetime, timedelta | from datetime import date, datetime, MINYEAR, timedelta | ||||||
| from json import loads | from json import loads | ||||||
| from os import path | from os import path | ||||||
| import secrets | import secrets | ||||||
| @@ -34,11 +34,11 @@ def _home(): | |||||||
|         write('system.log', f'Database error when processing request \'{request.url}\': {exception}') |         write('system.log', f'Database error when processing request \'{request.url}\': {exception}') | ||||||
|         return abort(500) |         return abort(500) | ||||||
|     current_tests = [ test for test in tests if test.end_date >= datetime.now() and test.start_date.date() <= date.today() ] |     current_tests = [ test for test in tests if test.end_date >= datetime.now() and test.start_date.date() <= date.today() ] | ||||||
|     current_tests.sort(key= lambda x: x.end_date, reverse=True) |     current_tests.sort(key= lambda x: x.end_date or datetime(MINYEAR,1,1), reverse=True) | ||||||
|     upcoming_tests = [ test for test in tests if test.start_date.date() > datetime.now().date()] |     upcoming_tests = [ test for test in tests if test.start_date.date() > datetime.now().date()] | ||||||
|     upcoming_tests.sort(key= lambda x: x.start_date) |     upcoming_tests.sort(key= lambda x: x.start_date or datetime(MINYEAR,1,1)) | ||||||
|     recent_results = [result for result in results if not result.status == 'started' ] |     recent_results = [result for result in results if not result.status == 'started' ] | ||||||
|     recent_results.sort(key= lambda x: x.end_time, reverse=True) |     recent_results.sort(key= lambda x: x.end_time or datetime(MINYEAR,1,1), reverse=True) | ||||||
|     return render_template('/admin/index.html', current_tests = current_tests, upcomimg_tests = upcoming_tests, recent_results = recent_results) |     return render_template('/admin/index.html', current_tests = current_tests, upcomimg_tests = upcoming_tests, recent_results = recent_results) | ||||||
|  |  | ||||||
| @admin.route('/settings/') | @admin.route('/settings/') | ||||||
| @@ -309,7 +309,7 @@ def _tests(filter:str=None): | |||||||
|     if filter in [None, '', 'active']: |     if filter in [None, '', 'active']: | ||||||
|         tests = [ test for test in _tests if test.end_date >= now and test.start_date <= now ] |         tests = [ test for test in _tests if test.end_date >= now and test.start_date <= now ] | ||||||
|         display_title = 'Active Exams' |         display_title = 'Active Exams' | ||||||
|         error_none = 'There are no exams that are currently active. You can create one using the Creat Exam form.' |         error_none = 'There are no exams that are currently active. You can create one using the Create Exam form.' | ||||||
|     if filter == 'expired': |     if filter == 'expired': | ||||||
|         tests = [ test for test in _tests if test.end_date < now ] |         tests = [ test for test in _tests if test.end_date < now ] | ||||||
|         display_title = 'Expired Exams' |         display_title = 'Expired Exams' | ||||||
|   | |||||||
| @@ -37,7 +37,7 @@ class Config(object): | |||||||
|         MYSQL_USER = os.getenv('MYSQL_USER') |         MYSQL_USER = os.getenv('MYSQL_USER') | ||||||
|         MYSQL_PASSWORD = os.getenv('MYSQL_PASSWORD') |         MYSQL_PASSWORD = os.getenv('MYSQL_PASSWORD') | ||||||
|         SQLALCHEMY_DATABASE_URI = f'mysql+pymysql://{MYSQL_USER}:{MYSQL_PASSWORD}@{DATABASE_HOST}:{DATABASE_PORT}/{MYSQL_DATABASE}' |         SQLALCHEMY_DATABASE_URI = f'mysql+pymysql://{MYSQL_USER}:{MYSQL_PASSWORD}@{DATABASE_HOST}:{DATABASE_PORT}/{MYSQL_DATABASE}' | ||||||
|     else: SQLALCHEMY_DATABASE_URI = f'sqlite:///{Path(os.path.abspath(f"{DATA}/database.db"))}' |     else: SQLALCHEMY_DATABASE_URI = f'sqlite:///{Path(os.path.abspath(f"{DATA}/db.sqlite"))}' | ||||||
|  |  | ||||||
| class Production(Config): | class Production(Config): | ||||||
|     pass |     pass | ||||||
|   | |||||||
| @@ -1,2 +1,2 @@ | |||||||
| <p>This web app was developed by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | <p>This web app was developed and is maintained by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | ||||||
| <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | ||||||
|     <div class="container"> |     <div class="container"> | ||||||
|         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest (Beta) | Admin</a> |         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest | Admin</a> | ||||||
|         <button |         <button | ||||||
|             class="navbar-toggler" |             class="navbar-toggler" | ||||||
|             type="button" |             type="button" | ||||||
|   | |||||||
| @@ -14,8 +14,7 @@ from pathlib import Path | |||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
|  |  | ||||||
| class Dataset(db.Model): | class Dataset(db.Model): | ||||||
|  |     id = db.Column(db.String(36), index=True, primary_key=True) | ||||||
|     id = db.Column(db.String(36), primary_key=True) |  | ||||||
|     name = db.Column(db.String(128), nullable=False) |     name = db.Column(db.String(128), nullable=False) | ||||||
|     tests = db.relationship('Test', backref='dataset') |     tests = db.relationship('Test', backref='dataset') | ||||||
|     creator_id = db.Column(db.String(36), db.ForeignKey('user.id')) |     creator_id = db.Column(db.String(36), db.ForeignKey('user.id')) | ||||||
|   | |||||||
| @@ -11,16 +11,15 @@ from datetime import datetime, timedelta | |||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
|  |  | ||||||
| class Entry(db.Model): | class Entry(db.Model): | ||||||
|  |     id = db.Column(db.String(36), index=True, primary_key=True) | ||||||
|     id = db.Column(db.String(36), primary_key=True) |  | ||||||
|     first_name = db.Column(db.String(128), nullable=False) |     first_name = db.Column(db.String(128), nullable=False) | ||||||
|     surname = db.Column(db.String(128), nullable=False) |     surname = db.Column(db.String(128), nullable=False) | ||||||
|     email = db.Column(db.String(128), nullable=False) |     email = db.Column(db.String(128), nullable=False) | ||||||
|     club = db.Column(db.String(128), nullable=True) |     club = db.Column(db.String(128), nullable=True) | ||||||
|     test_id = db.Column(db.String(36), db.ForeignKey('test.id')) |     test_id = db.Column(db.String(36), db.ForeignKey('test.id')) | ||||||
|     user_code = db.Column(db.String(6), nullable=True) |     user_code = db.Column(db.String(6), nullable=True) | ||||||
|     start_time = db.Column(db.DateTime, nullable=True) |     start_time = db.Column(db.DateTime, index=True, nullable=True) | ||||||
|     end_time = db.Column(db.DateTime, nullable=True) |     end_time = db.Column(db.DateTime, index=True, nullable=True) | ||||||
|     status = db.Column(db.String(16), nullable=True) |     status = db.Column(db.String(16), nullable=True) | ||||||
|     valid = db.Column(db.Boolean, default=True, nullable=True) |     valid = db.Column(db.Boolean, default=True, nullable=True) | ||||||
|     answers = db.Column(MutableJson, nullable=True) |     answers = db.Column(MutableJson, nullable=True) | ||||||
|   | |||||||
| @@ -9,10 +9,9 @@ import secrets | |||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
|  |  | ||||||
| class Test(db.Model): | class Test(db.Model): | ||||||
|      |     id = db.Column(db.String(36), index=True, primary_key=True) | ||||||
|     id = db.Column(db.String(36), primary_key=True) |     code = db.Column(db.String(36), index=True, nullable=False) | ||||||
|     code = db.Column(db.String(36), nullable=False) |     start_date = db.Column(db.DateTime, index=True, nullable=True) | ||||||
|     start_date = db.Column(db.DateTime, nullable=True) |  | ||||||
|     end_date = db.Column(db.DateTime, nullable=True) |     end_date = db.Column(db.DateTime, nullable=True) | ||||||
|     time_limit = db.Column(db.Integer, nullable=True) |     time_limit = db.Column(db.Integer, nullable=True) | ||||||
|     creator_id = db.Column(db.String(36), db.ForeignKey('user.id')) |     creator_id = db.Column(db.String(36), db.ForeignKey('user.id')) | ||||||
|   | |||||||
| @@ -11,11 +11,11 @@ from werkzeug.security import check_password_hash, generate_password_hash | |||||||
| import secrets | import secrets | ||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
| class User(UserMixin, db.Model): | class User(UserMixin, db.Model): | ||||||
|     id = db.Column(db.String(36), primary_key=True) |     id = db.Column(db.String(36), index=True, primary_key=True) | ||||||
|     username = db.Column(db.String(128), nullable=False) |     username = db.Column(db.String(128), nullable=False) | ||||||
|     password = db.Column(db.String(128), nullable=False) |     password = db.Column(db.String(128), nullable=False) | ||||||
|     email = db.Column(db.String(128), nullable=False) |     email = db.Column(db.String(128), nullable=False) | ||||||
|     reset_token = db.Column(db.String(20), nullable=True) |     reset_token = db.Column(db.String(20), index=True, nullable=True) | ||||||
|     verification_token = db.Column(db.String(20), nullable=True) |     verification_token = db.Column(db.String(20), nullable=True) | ||||||
|     tests = db.relationship('Test', backref='creator') |     tests = db.relationship('Test', backref='creator') | ||||||
|     datasets = db.relationship('Dataset', backref='creator') |     datasets = db.relationship('Dataset', backref='creator') | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ | |||||||
|         /> |         /> | ||||||
|         {% block style %} |         {% block style %} | ||||||
|         {% endblock %} |         {% endblock %} | ||||||
|         <title>{% block title %} SKA Referee Test Beta {% endblock %}</title> |         <title>{% block title %} SKA Referee Test {% endblock %}</title> | ||||||
|         {% include "quiz/components/og-meta.html" %} |         {% include "quiz/components/og-meta.html" %} | ||||||
|     </head> |     </head> | ||||||
|     <body class="bg-light"> |     <body class="bg-light"> | ||||||
| @@ -56,6 +56,8 @@ | |||||||
|             integrity="sha384-QJHtvGhmr9XOIpI6YVutG+2QOK9T+ZnN4kzFN1RtK3zEFEIsxhlmWl5/YESvpZ13" |             integrity="sha384-QJHtvGhmr9XOIpI6YVutG+2QOK9T+ZnN4kzFN1RtK3zEFEIsxhlmWl5/YESvpZ13" | ||||||
|             crossorigin="anonymous" |             crossorigin="anonymous" | ||||||
|         ></script> |         ></script> | ||||||
|  |         <!-- jQuery UI --> | ||||||
|  |         <script src="https://code.jquery.com/ui/1.13.2/jquery-ui.js"></script> | ||||||
|         <!-- Custom js --> |         <!-- Custom js --> | ||||||
|         <script type="text/javascript"> |         <script type="text/javascript"> | ||||||
|             var csrf_token = "{{ csrf_token() }}"; |             var csrf_token = "{{ csrf_token() }}"; | ||||||
|   | |||||||
| @@ -1,3 +1,3 @@ | |||||||
| <p>This web app was developed by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | <p>This web app was developed and is maintained by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | ||||||
| <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | ||||||
| <p>OpenDyslexic 3 is an open source typeface created by Abbie Gonzalez, licensed under a <a href="https://scripts.sil.org/OFL">SIL-OFL</a>. More information about OpenDyslexic is available <a href="https://opendyslexic.org/">on the project web site</a>.</p> | <p>OpenDyslexic 3 is an open source typeface created by Abbie Gonzalez, licensed under a <a href="https://scripts.sil.org/OFL">SIL-OFL</a>. More information about OpenDyslexic is available <a href="https://opendyslexic.org/">on the project web site</a>.</p> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark" id="primary-nav"> | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark" id="primary-nav"> | ||||||
|     <div class="container"> |     <div class="container"> | ||||||
|         <p class="navbar-brand mb-0 h1">SKA Refereeing Test (Beta)</p> |         <p class="navbar-brand mb-0 h1">SKA Refereeing Test</p> | ||||||
|         <div class="quiz-console w-100" style="display: none;" id="q-topbar"> |         <div class="quiz-console w-100" style="display: none;" id="q-topbar"> | ||||||
|             <div class="d-flex justify-content align-middle"> |             <div class="d-flex justify-content align-middle"> | ||||||
|                 <div class="container d-flex justify-content-center"> |                 <div class="container d-flex justify-content-center"> | ||||||
|   | |||||||
| @@ -1,6 +1,10 @@ | |||||||
| {% extends "quiz/components/base.html" %} | {% extends "quiz/components/base.html" %} | ||||||
| {% import "bootstrap/wtf.html" as wtf %} | {% import "bootstrap/wtf.html" as wtf %} | ||||||
|  |  | ||||||
|  | {% block style %} | ||||||
|  |     <link rel="stylesheet" href="https://code.jquery.com/ui/1.13.2/themes/base/jquery-ui.css"> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|     <div class="form-container"> |     <div class="form-container"> | ||||||
|         <form name="form-quiz-start" class="form-quiz-start"> |         <form name="form-quiz-start" class="form-quiz-start"> | ||||||
| @@ -43,4 +47,14 @@ | |||||||
|             </div> |             </div> | ||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
|  | {% endblock %} | ||||||
|  | {% block script %} | ||||||
|  |     <script> | ||||||
|  |         $( function() { | ||||||
|  |             const clubs = {{ clubs|tojson }} | ||||||
|  |             $('#club').autocomplete({ | ||||||
|  |                 source: clubs | ||||||
|  |             }) | ||||||
|  |         } ) | ||||||
|  |     </script> | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -29,6 +29,23 @@ def _instructions(): | |||||||
|  |  | ||||||
| @quiz.route('/start/', methods=['GET', 'POST']) | @quiz.route('/start/', methods=['GET', 'POST']) | ||||||
| def _start(): | def _start(): | ||||||
|  |     clubs = [ | ||||||
|  |         'Dundee Korfball Club', | ||||||
|  |         'Edinburgh City Korfball Club', | ||||||
|  |         'Edinburgh Mavericks Korfball Club', | ||||||
|  |         'Edinburgh University Korfball Club', | ||||||
|  |         'Glasgow Korfball Club', | ||||||
|  |         'Saint Andrews University Korfball Club', | ||||||
|  |         'Strathclyde University Korfball Club' | ||||||
|  |     ] | ||||||
|  |     try: entries = Entry.query.all() | ||||||
|  |     except Exception as exception: | ||||||
|  |         write('system.log', f'Database error when processing request \'{request.url}\': {exception}') | ||||||
|  |         return abort(500) | ||||||
|  |     for entry in entries: clubs.append(entry.get_club()) | ||||||
|  |     clubs = list(set(clubs)) | ||||||
|  |     try: clubs.remove('') | ||||||
|  |     except: pass | ||||||
|     form = StartQuiz() |     form = StartQuiz() | ||||||
|     if request.method == 'POST': |     if request.method == 'POST': | ||||||
|         if form.validate_on_submit(): |         if form.validate_on_submit(): | ||||||
| @@ -58,7 +75,7 @@ def _start(): | |||||||
|                 }), 200 |                 }), 200 | ||||||
|             return jsonify({'error': 'There was an error processing the user test and/or user codes.'}), 400 |             return jsonify({'error': 'There was an error processing the user test and/or user codes.'}), 400 | ||||||
|         return send_errors_to_client(form=form) |         return send_errors_to_client(form=form) | ||||||
|     return render_template('/quiz/start_quiz.html', form = form) |     return render_template('/quiz/start_quiz.html', form = form, clubs = clubs) | ||||||
|  |  | ||||||
| @quiz.route('/quiz/') | @quiz.route('/quiz/') | ||||||
| def _quiz(): | def _quiz(): | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ | |||||||
|         <link rel="shortcut icon" href="{{ url_for('views.static', filename='favicon.ico') }}"> |         <link rel="shortcut icon" href="{{ url_for('views.static', filename='favicon.ico') }}"> | ||||||
|         {% block style %} |         {% block style %} | ||||||
|         {% endblock %} |         {% endblock %} | ||||||
|         <title>{% block title %} SKA Referee Test Beta {% endblock %}</title> |         <title>{% block title %} SKA Referee Test {% endblock %}</title> | ||||||
|     </head> |     </head> | ||||||
|     <body class="bg-light"> |     <body class="bg-light"> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ | |||||||
|         /> |         /> | ||||||
|         {% block style %} |         {% block style %} | ||||||
|         {% endblock %} |         {% endblock %} | ||||||
|         <title>{% block title %} SKA Referee Test Beta {% endblock %}</title> |         <title>{% block title %} SKA Referee Test {% endblock %}</title> | ||||||
|         {% include "components/og-meta.html" %} |         {% include "components/og-meta.html" %} | ||||||
|     </head> |     </head> | ||||||
|     <body class="bg-light"> |     <body class="bg-light"> | ||||||
|   | |||||||
| @@ -1,3 +1,3 @@ | |||||||
| <p>This web app was developed by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | <p>This web app was developed and is maintained by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | ||||||
| <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | ||||||
| <p>OpenDyslexic 3 is an open source typeface created by Abbie Gonzalez, licensed under a <a href="https://scripts.sil.org/OFL">SIL-OFL</a>. More information about OpenDyslexic is available <a href="https://opendyslexic.org/">on the project web site</a>.</p> | <p>OpenDyslexic 3 is an open source typeface created by Abbie Gonzalez, licensed under a <a href="https://scripts.sil.org/OFL">SIL-OFL</a>. More information about OpenDyslexic is available <a href="https://opendyslexic.org/">on the project web site</a>.</p> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark" id="primary-nav"> | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark" id="primary-nav"> | ||||||
|     <div class="container"> |     <div class="container"> | ||||||
|         <p class="navbar-brand mb-0 h1">SKA Refereeing Test (Beta)</p> |         <p class="navbar-brand mb-0 h1">SKA Refereeing Test</p> | ||||||
|         <div class="quiz-console w-100" style="display: none;" id="q-topbar"> |         <div class="quiz-console w-100" style="display: none;" id="q-topbar"> | ||||||
|             <div class="d-flex justify-content align-middle"> |             <div class="d-flex justify-content align-middle"> | ||||||
|                 <div class="container d-flex justify-content-center"> |                 <div class="container d-flex justify-content-center"> | ||||||
|   | |||||||
| @@ -3,9 +3,19 @@ | |||||||
| {% block content %} | {% block content %} | ||||||
|     <h1>Privacy Policy</h1> |     <h1>Privacy Policy</h1> | ||||||
|  |  | ||||||
|     This web app stores data using cookies. The web site only stores the minimum information it needs to function. |     <ul> | ||||||
|     <h5>Site Administrators</h5> |         <li> | ||||||
|  |             This web app stores data using cookies. The web site only stores the minimum information it needs to function. | ||||||
|  |         </li> | ||||||
|  |         <li> | ||||||
|  |             All data stored on this app can be accessed by the SKA Committee and the maintainer of this app. | ||||||
|  |         </li> | ||||||
|  |         <li> | ||||||
|  |             This app is currently maintained by Vivek Santayana, a member of the Edinburgh City Korfball Club, with the permission of the SKA Committee. | ||||||
|  |         </li> | ||||||
|  |     </ul> | ||||||
|  |  | ||||||
|  |     <h5>Site Administrators</h5> | ||||||
|     <ul> |     <ul> | ||||||
|         <li>For site administrators, this web site uses encrypted cookies to store data from your log-in session.</li> |         <li>For site administrators, this web site uses encrypted cookies to store data from your log-in session.</li> | ||||||
|         <li>User information for administrators is encrypted and stored in a secure database, and are expunged when an account is deleted.</li> |         <li>User information for administrators is encrypted and stored in a secure database, and are expunged when an account is deleted.</li> | ||||||
| @@ -13,14 +23,14 @@ | |||||||
|      |      | ||||||
|     <h5>Test Candidates</h5> |     <h5>Test Candidates</h5> | ||||||
|     <ul> |     <ul> | ||||||
|         <li>The web site will not be trackin your log in, and all information about your test attempt will be stored on your device until you submit it to the server.</li> |         <li>The web site will not be tracking your log in, and all information about your test attempt will be stored on your device until you submit it to the server.</li> | ||||||
|         <li>Data from your test, including identifying information such as your name and email address, will be recorded by the Scottish Korfball Association in order to oversee the training and qualification of referees.</li> |         <li>Data from your test, including identifying information such as your name and email address, will be recorded by the Scottish Korfball Association in order to oversee the training and qualification of referees.</li> | ||||||
|         <li>These records will be kept for three years or until the expiration of the theory exam qualification (whichever is later), and will be expunged securely thereafter.</li> |         <li>These records will be kept for three years or until the expiration of the theory exam qualification (whichever is later), and will be expunged securely thereafter.</li> | ||||||
|         <li>All identifying information about candidates will be encrypted and stored in a secure database.</li> |         <li>All identifying information about candidates will be encrypted and stored in a secure database administered by the maintainer of this app.</li> | ||||||
|     </ul> |     </ul> | ||||||
|  |  | ||||||
|     <h5>Requests to Delete Data</h5> |     <h5>Requests to Delete Data</h5> | ||||||
|     <ul> |     <ul> | ||||||
|         <li>You can request to have any of your data that is held here deleted by emailing <a href="mailto:refereeing@scotlandkorfball.co.uk">refereeing@scotlandkorfball.co.uk</a>.</li> |         <li>You can request to view or delete data that the app stores about you by emailing <a href="mailto:refereeing@scotlandkorfball.co.uk">refereeing@scotlandkorfball.co.uk</a>.</li> | ||||||
|     </ul> |     </ul> | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -1,4 +1,3 @@ | |||||||
| from .data import load |  | ||||||
| from ..models import User | from ..models import User | ||||||
| from ..tools.logs import write | from ..tools.logs import write | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
|  |  | ||||||
| from ..extensions import db |  | ||||||
| from ..tools.logs import write | from ..tools.logs import write | ||||||
|  |  | ||||||
| from flask import jsonify | from flask import jsonify | ||||||
|   | |||||||
| @@ -1,2 +1,2 @@ | |||||||
| <p>This web app was developed by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | <p>This web app was developed and is maintained by Vivek Santayana. The source code for the web app, excluding any data pertaining to the questions in the quiz, is freely available at <a href="https://git.vsnt.uk/viveksantayana/ska-referee-test">Vivek’s personal GIT repository</a> under an MIT License.</p> | ||||||
| <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | <p>All questions in the test are © The Scottish Korfball Association {{ now.year }}. All rights are reserved.</p> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | <nav class="navbar fixed-top navbar-expand-md navbar-dark bg-dark"> | ||||||
|     <div class="container"> |     <div class="container"> | ||||||
|         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest (Beta) | Admin</a> |         <a href="{{ url_for('admin._home') }}" class="navbar-brand mb-0 h1">RefTest | Admin</a> | ||||||
|         <button |         <button | ||||||
|             class="navbar-toggler" |             class="navbar-toggler" | ||||||
|             type="button" |             type="button" | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| blinker==1.5 | blinker==1.5 | ||||||
| cffi==1.15.1 | cffi==1.15.1 | ||||||
| click==8.1.3 | click==8.1.3 | ||||||
| cryptography==37.0.4 | cryptography==38.0.1 | ||||||
| dnspython==2.2.1 | dnspython==2.2.1 | ||||||
| dominate==2.7.0 | dominate==2.7.0 | ||||||
| email-validator==1.2.1 | email-validator==1.2.1 | ||||||
| @@ -11,19 +11,22 @@ Flask-Login==0.6.2 | |||||||
| Flask-Mail==0.9.1 | Flask-Mail==0.9.1 | ||||||
| Flask-SQLAlchemy==2.5.1 | Flask-SQLAlchemy==2.5.1 | ||||||
| Flask-WTF==1.0.1 | Flask-WTF==1.0.1 | ||||||
| greenlet==1.1.2 | greenlet==1.1.3 | ||||||
| gunicorn==20.1.0 | gunicorn==20.1.0 | ||||||
| idna==3.3 | idna==3.3 | ||||||
| itsdangerous==2.1.2 | itsdangerous==2.1.2 | ||||||
| Jinja2==3.1.2 | Jinja2==3.1.2 | ||||||
| MarkupSafe==2.1.1 | MarkupSafe==2.1.1 | ||||||
|  | pip==22.2.2 | ||||||
| pycparser==2.21 | pycparser==2.21 | ||||||
| PyMySQL==1.0.2 | PyMySQL==1.0.2 | ||||||
| python-dotenv==0.20.0 | python-dotenv==0.21.0 | ||||||
|  | setuptools==65.3.0 | ||||||
| six==1.16.0 | six==1.16.0 | ||||||
| SQLAlchemy==1.4.40 | SQLAlchemy==1.4.41 | ||||||
| sqlalchemy-json==0.5.0 | sqlalchemy-json==0.5.0 | ||||||
| SQLAlchemy-Utils==0.38.3 | SQLAlchemy-Utils==0.38.3 | ||||||
| visitor==0.1.3 | visitor==0.1.3 | ||||||
| Werkzeug==2.2.2 | Werkzeug==2.2.2 | ||||||
|  | wheel==0.37.1 | ||||||
| WTForms==3.0.1 | WTForms==3.0.1 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user