Commit ab185529f9f3b53d77631bf6835299267ca7c531
1 parent
43c0279e
Exists in
master
and in
1 other branch
Changes:
- initdb.py sorts users by id - start_topic generates questions in threads - add sanity_check_questions as command line option `--check`
Showing
7 changed files
with
90 additions
and
51 deletions
Show diff stats
aprendizations/factory.py
| ... | ... | @@ -35,7 +35,8 @@ from aprendizations.tools import run_script |
| 35 | 35 | from aprendizations.questions import (QuestionInformation, QuestionRadio, |
| 36 | 36 | QuestionCheckbox, QuestionText, |
| 37 | 37 | QuestionTextRegex, QuestionTextArea, |
| 38 | - QuestionNumericInterval) | |
| 38 | + QuestionNumericInterval, | |
| 39 | + QuestionException) | |
| 39 | 40 | |
| 40 | 41 | # setup logger for this module |
| 41 | 42 | logger = logging.getLogger(__name__) |
| ... | ... | @@ -68,10 +69,6 @@ class QFactory(object): |
| 68 | 69 | # Given a ref returns an instance of a descendent of Question(), |
| 69 | 70 | # i.e. a question object (radio, checkbox, ...). |
| 70 | 71 | # ----------------------------------------------------------------------- |
| 71 | - # async def generate_async(self): | |
| 72 | - # loop = asyncio.get_running_loop() | |
| 73 | - # return await loop.run_in_executor(None, self.generate) | |
| 74 | - | |
| 75 | 72 | def generate(self): |
| 76 | 73 | logger.debug(f'Generating "{self.question["ref"]}"...') |
| 77 | 74 | # Shallow copy so that script generated questions will not replace |
| ... | ... | @@ -92,8 +89,11 @@ class QFactory(object): |
| 92 | 89 | # Finally we create an instance of Question() |
| 93 | 90 | try: |
| 94 | 91 | qinstance = self._types[q['type']](q) # instance matching class |
| 95 | - except KeyError as e: | |
| 96 | - logger.error(f'Unknown type "{q["type"]}" in "{q["ref"]}"') | |
| 92 | + except QuestionException as e: | |
| 93 | + logger.error(e) | |
| 97 | 94 | raise e |
| 95 | + except KeyError: | |
| 96 | + logger.error(f'Invalid type "{q["type"]}" in "{q["ref"]}"') | |
| 97 | + raise | |
| 98 | 98 | else: |
| 99 | 99 | return qinstance | ... | ... |
aprendizations/initdb.py
| ... | ... | @@ -105,7 +105,6 @@ def insert_students_into_db(session, students): |
| 105 | 105 | # --- start db session --- |
| 106 | 106 | session.add_all([Student(id=s['uid'], name=s['name'], password=s['pw']) |
| 107 | 107 | for s in students]) |
| 108 | - | |
| 109 | 108 | session.commit() |
| 110 | 109 | |
| 111 | 110 | except sa.exc.IntegrityError: |
| ... | ... | @@ -116,7 +115,7 @@ def insert_students_into_db(session, students): |
| 116 | 115 | # =========================================================================== |
| 117 | 116 | def show_students_in_database(session, verbose=False): |
| 118 | 117 | try: |
| 119 | - users = session.query(Student).order_by(Student.id).all() | |
| 118 | + users = session.query(Student).all() | |
| 120 | 119 | except Exception: |
| 121 | 120 | raise |
| 122 | 121 | else: |
| ... | ... | @@ -125,6 +124,7 @@ def show_students_in_database(session, verbose=False): |
| 125 | 124 | if n == 0: |
| 126 | 125 | print(' -- none --') |
| 127 | 126 | else: |
| 127 | + users.sort(key=lambda u: f'{u.id:>12}') # sort by number | |
| 128 | 128 | if verbose: |
| 129 | 129 | for u in users: |
| 130 | 130 | print(f'{u.id:>12} {u.name}') |
| ... | ... | @@ -162,8 +162,7 @@ def main(): |
| 162 | 162 | # --- password hashing |
| 163 | 163 | if students: |
| 164 | 164 | print(f'Generating password hashes', end='') |
| 165 | - def hash_func(s): return hashpw(s, args.pw) | |
| 166 | - with ThreadPoolExecutor() as executor: # hashing in parallel | |
| 165 | + with ThreadPoolExecutor() as executor: | |
| 167 | 166 | executor.map(lambda s: hashpw(s, args.pw), students) |
| 168 | 167 | print() |
| 169 | 168 | ... | ... |
aprendizations/knowledge.py
| ... | ... | @@ -3,6 +3,7 @@ |
| 3 | 3 | import random |
| 4 | 4 | from datetime import datetime |
| 5 | 5 | import logging |
| 6 | +import asyncio | |
| 6 | 7 | |
| 7 | 8 | # libraries |
| 8 | 9 | import networkx as nx |
| ... | ... | @@ -73,7 +74,7 @@ class StudentKnowledge(object): |
| 73 | 74 | async def start_topic(self, topic): |
| 74 | 75 | logger.debug('StudentKnowledge.start_topic()') |
| 75 | 76 | if self.current_topic == topic: |
| 76 | - logger.debug(' Restarting current topic is not allowed.') | |
| 77 | + logger.info(' Restarting current topic is not allowed.') | |
| 77 | 78 | return False |
| 78 | 79 | |
| 79 | 80 | # do not allow locked topics |
| ... | ... | @@ -95,9 +96,12 @@ class StudentKnowledge(object): |
| 95 | 96 | logger.debug(f'Questions: {", ".join(questions)}') |
| 96 | 97 | |
| 97 | 98 | # generate instances of questions |
| 98 | - # gen = lambda qref: self.factory[qref].generate() | |
| 99 | - self.questions = [self.factory[qref].generate() for qref in questions] | |
| 100 | - # self.questions = [gen(qref) for qref in questions] | |
| 99 | + # self.questions = [self.factory[ref].generate() for ref in questions] | |
| 100 | + loop = asyncio.get_running_loop() | |
| 101 | + generators = [loop.run_in_executor(None, self.factory[qref].generate) | |
| 102 | + for qref in questions] | |
| 103 | + self.questions = await asyncio.gather(*generators) | |
| 104 | + | |
| 101 | 105 | logger.debug(f'Total: {len(self.questions)} questions') |
| 102 | 106 | |
| 103 | 107 | # get first question |
| ... | ... | @@ -117,6 +121,7 @@ class StudentKnowledge(object): |
| 117 | 121 | 'level': self.correct_answers / (self.correct_answers + |
| 118 | 122 | self.wrong_answers) |
| 119 | 123 | } |
| 124 | + # self.current_topic = None | |
| 120 | 125 | self.unlock_topics() |
| 121 | 126 | |
| 122 | 127 | # ------------------------------------------------------------------------ | ... | ... |
aprendizations/learnapp.py
| ... | ... | @@ -60,7 +60,9 @@ class LearnApp(object): |
| 60 | 60 | session.close() |
| 61 | 61 | |
| 62 | 62 | # ------------------------------------------------------------------------ |
| 63 | - def __init__(self, config_files, prefix, db): | |
| 63 | + # init | |
| 64 | + # ------------------------------------------------------------------------ | |
| 65 | + def __init__(self, config_files, prefix, db, check=False): | |
| 64 | 66 | self.db_setup(db) # setup database and check students |
| 65 | 67 | self.online = dict() # online students |
| 66 | 68 | |
| ... | ... | @@ -68,9 +70,22 @@ class LearnApp(object): |
| 68 | 70 | for c in config_files: |
| 69 | 71 | self.populate_graph(c) |
| 70 | 72 | |
| 71 | - self.build_factory() # for all questions of all topics | |
| 73 | + self.factory = self.make_factory() # for all questions all topics | |
| 72 | 74 | self.db_add_missing_topics(self.deps.nodes()) |
| 73 | 75 | |
| 76 | + if check: | |
| 77 | + self.sanity_check_questions() | |
| 78 | + | |
| 79 | + # ------------------------------------------------------------------------ | |
| 80 | + def sanity_check_questions(self): | |
| 81 | + for qref, q in self.factory.items(): | |
| 82 | + logger.info(f'Generating {qref}...') | |
| 83 | + try: | |
| 84 | + q.generate() | |
| 85 | + except Exception as e: | |
| 86 | + logger.error(f'Sanity check failed in "{qref}"') | |
| 87 | + raise e | |
| 88 | + | |
| 74 | 89 | # ------------------------------------------------------------------------ |
| 75 | 90 | # login |
| 76 | 91 | # ------------------------------------------------------------------------ |
| ... | ... | @@ -196,8 +211,8 @@ class LearnApp(object): |
| 196 | 211 | student = self.online[uid]['state'] |
| 197 | 212 | try: |
| 198 | 213 | await student.start_topic(topic) |
| 199 | - except KeyError: | |
| 200 | - logger.warning(f'User "{uid}" opened nonexistent topic: "{topic}"') | |
| 214 | + except Exception: | |
| 215 | + logger.warning(f'User "{uid}" could not start topic "{topic}"') | |
| 201 | 216 | else: |
| 202 | 217 | logger.info(f'User "{uid}" started topic "{topic}"') |
| 203 | 218 | |
| ... | ... | @@ -267,7 +282,8 @@ class LearnApp(object): |
| 267 | 282 | t['name'] = attr.get('name', tref) |
| 268 | 283 | t['path'] = path.join(g.graph['prefix'], tref) # prefix/topic |
| 269 | 284 | t['file'] = attr.get('file', default_file) # questions.yaml |
| 270 | - t['shuffle_questions'] = attr.get('shuffle_questions', default_shuffle_questions) | |
| 285 | + t['shuffle_questions'] = attr.get('shuffle_questions', | |
| 286 | + default_shuffle_questions) | |
| 271 | 287 | t['max_tries'] = attr.get('max_tries', default_maxtries) |
| 272 | 288 | t['forgetting_factor'] = attr.get('forgetting_factor', |
| 273 | 289 | default_forgetting_factor) |
| ... | ... | @@ -278,12 +294,16 @@ class LearnApp(object): |
| 278 | 294 | |
| 279 | 295 | logger.info(f'Loaded {g.number_of_nodes()} topics') |
| 280 | 296 | |
| 297 | + # ======================================================================== | |
| 298 | + # methods that do not change state (pure functions) | |
| 299 | + # ======================================================================== | |
| 300 | + | |
| 281 | 301 | # ------------------------------------------------------------------------ |
| 282 | 302 | # Buils dictionary of question factories |
| 283 | 303 | # ------------------------------------------------------------------------ |
| 284 | - def build_factory(self): | |
| 304 | + def make_factory(self): | |
| 285 | 305 | logger.info('Building questions factory') |
| 286 | - self.factory = {} # {'qref': QFactory()} | |
| 306 | + factory = {} # {'qref': QFactory()} | |
| 287 | 307 | g = self.deps |
| 288 | 308 | for tref in g.nodes(): |
| 289 | 309 | t = g.node[tref] |
| ... | ... | @@ -310,15 +330,12 @@ class LearnApp(object): |
| 310 | 330 | |
| 311 | 331 | for q in questions: |
| 312 | 332 | if q['ref'] in t['questions']: |
| 313 | - self.factory[q['ref']] = QFactory(q) | |
| 333 | + factory[q['ref']] = QFactory(q) | |
| 314 | 334 | |
| 315 | 335 | logger.info(f'{len(t["questions"]):6} {tref}') |
| 316 | 336 | |
| 317 | - logger.info(f'Factory contains {len(self.factory)} questions') | |
| 318 | - | |
| 319 | - # ======================================================================== | |
| 320 | - # methods that do not change state (pure functions) | |
| 321 | - # ======================================================================== | |
| 337 | + logger.info(f'Factory contains {len(factory)} questions') | |
| 338 | + return factory | |
| 322 | 339 | |
| 323 | 340 | # ------------------------------------------------------------------------ |
| 324 | 341 | def get_login_counter(self, uid): | ... | ... |
aprendizations/questions.py
| ... | ... | @@ -13,6 +13,10 @@ from aprendizations.tools import run_script |
| 13 | 13 | logger = logging.getLogger(__name__) |
| 14 | 14 | |
| 15 | 15 | |
| 16 | +class QuestionException(Exception): | |
| 17 | + pass | |
| 18 | + | |
| 19 | + | |
| 16 | 20 | # =========================================================================== |
| 17 | 21 | # Questions derived from Question are already instantiated and ready to be |
| 18 | 22 | # presented to students. |
| ... | ... | @@ -63,12 +67,12 @@ class QuestionRadio(Question): |
| 63 | 67 | ''' |
| 64 | 68 | |
| 65 | 69 | # ------------------------------------------------------------------------ |
| 70 | + # FIXME marking all options right breaks | |
| 66 | 71 | def __init__(self, q): |
| 67 | 72 | super().__init__(q) |
| 68 | 73 | |
| 69 | 74 | n = len(self['options']) |
| 70 | 75 | |
| 71 | - # set defaults if missing | |
| 72 | 76 | self.set_defaults({ |
| 73 | 77 | 'text': '', |
| 74 | 78 | 'correct': 0, |
| ... | ... | @@ -76,32 +80,40 @@ class QuestionRadio(Question): |
| 76 | 80 | 'discount': True, |
| 77 | 81 | }) |
| 78 | 82 | |
| 79 | - # always convert to list, e.g. correct: 2 --> correct: [0,0,1,0,0] | |
| 83 | + # convert int to list, e.g. correct: 2 --> correct: [0,0,1,0,0] | |
| 80 | 84 | # correctness levels from 0.0 to 1.0 (no discount here!) |
| 81 | 85 | if isinstance(self['correct'], int): |
| 82 | 86 | self['correct'] = [1.0 if x == self['correct'] else 0.0 |
| 83 | 87 | for x in range(n)] |
| 84 | 88 | |
| 89 | + if len(self['correct']) != n: | |
| 90 | + msg = f'Options and correct mismatch in "{self["ref"]}"' | |
| 91 | + raise QuestionException(msg) | |
| 92 | + | |
| 85 | 93 | if self['shuffle']: |
| 86 | - # separate right from wrong options | |
| 87 | - right = [i for i in range(n) if self['correct'][i] == 1] | |
| 94 | + # lists with indices of right and wrong options | |
| 95 | + right = [i for i in range(n) if self['correct'][i] >= 1] | |
| 88 | 96 | wrong = [i for i in range(n) if self['correct'][i] < 1] |
| 89 | 97 | |
| 90 | 98 | self.set_defaults({'choose': 1+len(wrong)}) |
| 91 | 99 | |
| 92 | - # choose 1 correct option | |
| 93 | - r = random.choice(right) | |
| 94 | - options = [self['options'][r]] | |
| 95 | - correct = [1.0] | |
| 100 | + # try to choose 1 correct option | |
| 101 | + if right: | |
| 102 | + r = random.choice(right) | |
| 103 | + options = [self['options'][r]] | |
| 104 | + correct = [self['correct'][r]] | |
| 105 | + else: | |
| 106 | + options = [] | |
| 107 | + correct = [] | |
| 96 | 108 | |
| 97 | 109 | # choose remaining wrong options |
| 98 | - random.shuffle(wrong) | |
| 99 | - nwrong = self['choose']-1 | |
| 100 | - options.extend(self['options'][i] for i in wrong[:nwrong]) | |
| 101 | - correct.extend(self['correct'][i] for i in wrong[:nwrong]) | |
| 110 | + nwrong = self['choose'] - len(correct) | |
| 111 | + wrongsample = random.sample(wrong, k=nwrong) | |
| 112 | + options += [self['options'][i] for i in wrongsample] | |
| 113 | + correct += [self['correct'][i] for i in wrongsample] | |
| 102 | 114 | |
| 103 | 115 | # final shuffle of the options |
| 104 | - perm = random.sample(range(self['choose']), self['choose']) | |
| 116 | + perm = random.sample(range(self['choose']), k=self['choose']) | |
| 105 | 117 | self['options'] = [str(options[i]) for i in perm] |
| 106 | 118 | self['correct'] = [float(correct[i]) for i in perm] |
| 107 | 119 | |
| ... | ... | @@ -118,8 +130,6 @@ class QuestionRadio(Question): |
| 118 | 130 | x = (x - x_aver) / (1.0 - x_aver) |
| 119 | 131 | self['grade'] = x |
| 120 | 132 | |
| 121 | - return self['grade'] | |
| 122 | - | |
| 123 | 133 | |
| 124 | 134 | # =========================================================================== |
| 125 | 135 | class QuestionCheckbox(Question): |
| ... | ... | @@ -150,8 +160,8 @@ class QuestionCheckbox(Question): |
| 150 | 160 | }) |
| 151 | 161 | |
| 152 | 162 | if len(self['correct']) != n: |
| 153 | - logger.error(f'Options and correct size mismatch in ' | |
| 154 | - f'"{self["ref"]}", file "{self["filename"]}".') | |
| 163 | + msg = f'Options and correct mismatch in "{self["ref"]}"' | |
| 164 | + raise QuestionException(msg) | |
| 155 | 165 | |
| 156 | 166 | # if an option is a list of (right, wrong), pick one |
| 157 | 167 | # FIXME it's possible that all options are chosen wrong |
| ... | ... | @@ -168,9 +178,9 @@ class QuestionCheckbox(Question): |
| 168 | 178 | # generate random permutation, e.g. [2,1,4,0,3] |
| 169 | 179 | # and apply to `options` and `correct` |
| 170 | 180 | if self['shuffle']: |
| 171 | - perm = random.sample(range(n), self['choose']) | |
| 172 | - self['options'] = [options[i] for i in perm] | |
| 173 | - self['correct'] = [correct[i] for i in perm] | |
| 181 | + perm = random.sample(range(n), k=self['choose']) | |
| 182 | + self['options'] = [str(options[i]) for i in perm] | |
| 183 | + self['correct'] = [float(correct[i]) for i in perm] | |
| 174 | 184 | |
| 175 | 185 | # ------------------------------------------------------------------------ |
| 176 | 186 | # can return negative values for wrong answers |
| ... | ... | @@ -338,7 +348,10 @@ class QuestionTextArea(Question): |
| 338 | 348 | except KeyError: |
| 339 | 349 | logger.error(f'No grade in "{self["correct"]}".') |
| 340 | 350 | else: |
| 341 | - self['grade'] = float(out) | |
| 351 | + try: | |
| 352 | + self['grade'] = float(out) | |
| 353 | + except (TypeError, ValueError): | |
| 354 | + logger.error(f'Invalid grade in "{self["correct"]}".') | |
| 342 | 355 | |
| 343 | 356 | |
| 344 | 357 | # =========================================================================== | ... | ... |
aprendizations/serve.py
| ... | ... | @@ -375,6 +375,11 @@ def parse_cmdline_arguments(): |
| 375 | 375 | ) |
| 376 | 376 | |
| 377 | 377 | argparser.add_argument( |
| 378 | + '--check', action='store_true', | |
| 379 | + help='Sanity check all questions' | |
| 380 | + ) | |
| 381 | + | |
| 382 | + argparser.add_argument( | |
| 378 | 383 | '--debug', action='store_true', |
| 379 | 384 | help='Enable debug messages' |
| 380 | 385 | ) |
| ... | ... | @@ -451,7 +456,7 @@ def main(): |
| 451 | 456 | # --- start application |
| 452 | 457 | logging.info('Starting App') |
| 453 | 458 | try: |
| 454 | - learnapp = LearnApp(arg.conffile, prefix=arg.prefix, db=arg.db) | |
| 459 | + learnapp = LearnApp(arg.conffile, prefix=arg.prefix, db=arg.db, check=arg.check) | |
| 455 | 460 | except Exception as e: |
| 456 | 461 | logging.critical('Failed to start application') |
| 457 | 462 | raise e | ... | ... |
aprendizations/tools.py
| ... | ... | @@ -147,7 +147,7 @@ def load_yaml(filename, default=None): |
| 147 | 147 | else: |
| 148 | 148 | with f: |
| 149 | 149 | try: |
| 150 | - default = yaml.safe_load(f) # FIXME check if supports all kinds of questions including regex | |
| 150 | + default = yaml.safe_load(f) | |
| 151 | 151 | except yaml.YAMLError as e: |
| 152 | 152 | mark = e.problem_mark |
| 153 | 153 | logger.error(f'In file "{filename}" near line {mark.line}, ' | ... | ... |