Skip to content

Latest commit

 

History

History
344 lines (255 loc) · 13.5 KB

README.rst

File metadata and controls

344 lines (255 loc) · 13.5 KB

TEEPy

TEEPy stands for Tech Engineering Exam in Python. TEEPy's aim is to make an easy-to-use framework to create clean and professional-looking paper-based assessments (e.g., quiz, exam, etc.) with a focus on STEM (Science, Technology, Engineering, and Math) related topics. There are two intended end-users of TEEPy, question creators (QCs), and assessment creators (ACs). QCs create individual questions or problems that will be used in a particular assessment that an AC will use to create the overall assessment. TEEPy accomplishes these goals by generating all question and assessment content as HTML and then rendering the HTML to create PDFs (using a combination of cefpython and pyppeteer). There is also some Beautiful Soup thrown in there as well. TEEPy supports the use of LaTex (via MathJax) as well as the use of units (using the package pint). TEEPy is capable of randomizing question placement and answer choices to questions.

Installation

Installation is a simple matter of

$ pip install teepy

and then enjoy!

Roles

There are two roles that an end-user of TEEPy can assume, question creator (QC) or assessment creator (AC). QCs deal solely with individual questions or problem creation. ACs deal with the assessment as a whole. ACs take one or more questions created by QCs and make an assessment out of the selected questions.

Question Creator (QC)

The goal of a QC is to create individual questions that are self-contained. The content and calculations in the individual question file will not interact with content and calculations performed in another question file. Each question file must contain two function definitions, a PROBLEM() function and a CHOICES() function. Each of these functions takes one argument, an index value variable. The index value is used to create different versions of a particular question. The PROBLEM() function uses the function's docstring to define the problem statement and should return a dictionary containing at least a key of answer. If the question has no answer (e.g., in the case of an open-ended question), a None may be returned from the PROBLEM() function. An example of an open-ended question is shown below.

# An open-ended question
def PROBLEM(ind):
    '''What is the meaning of life?'''

    return None

Even if there are no different versions of the question, the PROBLEM() should be a function of the index variable ind.

In the event a question does have a correct answer (or answers), the value of the key answer should be a single value or a list of values (in the case of a multiple answer problem). An example of a single answer question is shown below.

# A single answer question
def PROBLEM(ind):
    '''What color is the sky?'''

    answer = 'Blue'

    return {'answer': answer}

An example of a multiple-answer question is shown below.

# A multiple-answer question
def PROBLEM(ind):
    '''How many licks does it take to get to the center of a Tootsie Pop?'''

    answer = [3, 'The world may never know.']

    return {'answer': answer}

The return value of PROBLEM() may also contain a key of given. The value of the key given should be a dictionary that includes any variables used in the question statement. An example of using a given variable is shown below.

# A question with a given variable
def PROBLEM(ind):
    '''A {object} is an example of what?'''

    obj = ['dog', 'carrot', 'diamond']

    answers = ['Animal', 'Vegatable', 'Mineral']

    given = {'object': obj[ind]}

    return {'answer': answers[ind], 'given': given}

It should be noted in the example above that three different versions of the questions may be created by simply changing the ind variable to a value of zero, one, or two. Units may also be used in the PROBLEM() function. An example of utilizing units is shown below.

import teepy

def PROBLEM(ind):
    '''If points A, B, and C lie along a straight line in that order,
and the distance between point A and B is $ {L1} $, and the distance
between point B and C is $ {L2} $, what is the distance between point
A and C?'''

    L1s = [1, 2, 3]
    L2s = [4, 5, 6]

    L1 = teepy.define_unit(L1s[ind], 'ft')
    L2 = teepy.define_unit(L2s[ind], 'cm')

    L = L1 + L2

    answer = L.to('m')
    given = {'L1': L1,
             'L2': L2}

    return {'answer': answer, 'given': given}

There are a few things to note about the example above. If a given variable has units, the rendered version of the variable (i.e., what is in the problem statement) needs to be enclosed in dollar signs. The units of a given variable get converted into LaTeX. LaTeX code that is not enclosed in dollar signs will not be rendered as LaTeX. The TEEPy function define_unit may be used to assign units to a variable. This function is pint's Q_ function (please refer to pint's documentation on how to use it). Once units have been assigned to a variable, calculations performed with those variables will automatically perform the necessary conversions when dealing with different types of units.

The CHOICES() function must return a None value, or a dictionary containing the key choices. No multiple-choice choices will be displayed if CHOICES() returns a None value. An open-ended question is typically when this is needed. Below is an example of a CHOICES() function that returns a None value.

def CHOICES(ind):

    return None

If multiple-choice answers are provided, the CHOICES() function should return a dictionary containing the key choices. The value of this key should be a list containing the correct answer and wrong answers. In other words, it should include everything that is to be listed as answer choices in the question. An example of using the choices key-value pair is shown below.

import teepy

def CHOICES(ind):
    choices = teepy.get_answers(PROBLEM(ind))

    choices.extend(['Red',
                    'Green',
                    'Yellow',
                    'Orange'])

    random.shuffle(choices)

    return {'choices': choices}

The example above also illustrates the use of a TEEPy function called get_answers(). The function takes one argument of a PROBLEMS() function with the particular index value that is to used. The function always returns a list even if the answer to the problem is a single value answer. The example above also demonstrates the use of Python's built-in module random. random has many useful methods but the one here shuffles a list. The list of choices does not have to be rearranged. An example of not mixing the list of options is shown below.

import teepy

def CHOICES(ind):
    choices = [1, 2]
    choices.extend(teepy.get_answers(PROBLEM(ind)))

    return {'choices': choices}

There are a couple of things worth mentioning about the CHOICES() function when an answer has units. When an answer has units, TEEPy has the function generate_choices() available to generate randomized choices. The function takes three arguments; the number of choices, the correct answer, and the step size between choices. The CHOICES() function must also have a key choice_format in the dictionary it returns. The value of this key is the desired format type of the answer choices. An example of using the generate_choices() function and the choice_format key is shown below.

import teepy

def CHOICES(ind):
    N = 10
    choice_format = '{:0.3f}'
    step = random.uniform(0.01, 0.05)
    ans = teepy.get_answers(PROBLEM(ind))

    choices = teepy.generate_choices(N, ans, step)

    return {'choices': choices, 'choice_format': choice_format}

All of the examples seen above may be found in the examples directory.

Assessment Creator (AC)

An assessment creator (AC) is an individual that takes questions created by QCs and arranges them to form an assessment (e.g., quiz, exam, etc.). The AC will design the assessment layout and set the question point values. To do so, an AC will initialize a teepy.begin() class. This class will be the handle for structuring the assessment content. When initializing the class, two arguments are typically used; n_forms and n_inds. n_forms sets the number of randomized forms to generate. This value is typically the number of students taking the assessment. The n_inds argument sets the number of versions each question has. Currently, TEEPy requires the number of versions a question has to be the same between different questions (i.e., if question A has three different versions, question B must also have three different versions). The exception to this rule is if the question's PROBLEM() and CHOICES() function do not utilize the ind variable (as in the case of the open-ended question shown previously). An example of initializing the TEEPy begin class is shown below.

import teepy

exam = teepy.begin(n_forms = 5, n_inds = 3)

In the example above, the TEEPy begin class will generate five randomized assessments where each question in the assessment has three different versions. The begin class has a method called HTML that allows the addition of arbitrary HTML content to be added to the assessment. An example of using the HTML is shown below.

exam.HTML('''<div style="text-align: center;">
<h1>COURSE NAME</h1><br>
<h2>ASSESMENT NAME</h2><br>
<h3>DATE</h3><br><br>
<h3>Form Number: ''' + exam.form_number() + '''</h3><br><br>
<h4>Printed Name: ________________________________________ </h4>
</div>''')

In the example above, the method form_number is implemented. This method inserts the randomly generated assessment form number. It is always important to include this on an assessment. Otherwise, assessment forms may not be distinguished from each other. Other methods are exposed in the begin class. An example of the new_page and problem methods are shown below.

exam.new_page()

exam.problem('path/to/question/file', 5)

The new_page method inserts a page break at its placement. The problem method is how question files are added to an assessment. The problem method requires two arguments; the path to the question file and the question's point value if a correct answer is given. The problem method also can accept the keyword arguments of display_worth, min_height, and pts_incorrect. display_worth is a Boolean that sets whether the question's point value should be displayed or not. min_height sets the minimum height of a question. The value supplied to min_height gets translated into the CSS property of min-height using units of inches (e.g., min_height = 1.5 means the minimum height of the problem will be 1.5 inches). This keyword argument is helpful in questions that require a certain amount of paper space for a student's written computation. The pts_incorrect keyword argument indicates that an incorrect answer should result in a deduction of points, not just the failure to earn points. An example of utilizing the various keyword arguments of the method problem is shown below.

exam.problem('path/to/question/file', 0,
             display_worth = False,
             min_height = 2.5,
             pts_incorrect = -2)

The final method exposed in the TEEPy begin class is the section method. This method allows for the grouping of content in the assessment. An example of its use is shown below.

concept_section = exam.section(shuffle = True)

concept_section.problem('path/to/concept/question1', 3)
concept_section.problem('path/to/concept/quesiton2', 3)

exam.section(concept_section)

The order of content in a section may be randomized by setting the keyword argument shuffle to True. The final step in creating an assessment is to generate the forms. The assessment forms are generated by invoking the generate method of the TEEPy begin class. Utilizing the generate method is shown below.

exam.generate()

Whenever the generate method is invoked, the number of assessment forms are generated, also well as n_inds reference forms (which are assessment forms with all shuffling disabled and with the correct answers marked), and an Excel sheet containing assessment key information (i.e., correct answers for a particular form, etc.). More examples of different functionality of TEEPy can be found in the examples directory.