57 lines
1.7 KiB
Python
57 lines
1.7 KiB
Python
from calc import evaluator, UndefinedVariable
|
|
|
|
#-----------------------------------------------------------------------------
|
|
#
|
|
# Utility functions used in CAPA responsetypes
|
|
|
|
|
|
def compare_with_tolerance(v1, v2, tol):
|
|
''' Compare v1 to v2 with maximum tolerance tol
|
|
tol is relative if it ends in %; otherwise, it is absolute
|
|
|
|
- v1 : student result (number)
|
|
- v2 : instructor result (number)
|
|
- tol : tolerance (string or number)
|
|
|
|
'''
|
|
relative = tol.endswith('%')
|
|
if relative:
|
|
tolerance_rel = evaluator(dict(), dict(), tol[:-1]) * 0.01
|
|
tolerance = tolerance_rel * max(abs(v1), abs(v2))
|
|
else:
|
|
tolerance = evaluator(dict(), dict(), tol)
|
|
return abs(v1 - v2) <= tolerance
|
|
|
|
|
|
def contextualize_text(text, context): # private
|
|
''' Takes a string with variables. E.g. $a+$b.
|
|
Does a substitution of those variables from the context '''
|
|
if not text: return text
|
|
for key in sorted(context, lambda x, y: cmp(len(y), len(x))):
|
|
text = text.replace('$' + key, str(context[key]))
|
|
return text
|
|
|
|
|
|
def convert_files_to_filenames(answers):
|
|
'''
|
|
Check for File objects in the dict of submitted answers,
|
|
convert File objects to their filename (string)
|
|
'''
|
|
new_answers = dict()
|
|
for answer_id in answers.keys():
|
|
if is_file(answers[answer_id]):
|
|
new_answers[answer_id] = answers[answer_id].name
|
|
else:
|
|
new_answers[answer_id] = answers[answer_id]
|
|
return new_answers
|
|
|
|
def is_file(file_to_test):
|
|
'''
|
|
Duck typing to check if 'file_to_test' is a File object
|
|
'''
|
|
is_file = True
|
|
for method in ['read', 'name']:
|
|
if not hasattr(file_to_test, method):
|
|
is_file = False
|
|
return is_file
|