~azzar1/unity/add-show-desktop-key

« back to all changes in this revision

Viewing changes to ivle/worksheet.py

  • Committer: William Grant
  • Date: 2009-03-04 03:47:10 UTC
  • Revision ID: grantw@unimelb.edu.au-20090304034710-lxyq8gggryfuwqbp
Only redirect +login for logged in users if the method is not POST. We do this
to allow people to use their browser's history to get to a login form, and log
in again. It's also useful for strange people who like to send POST requests
themselves.

Fixes Google Code issue 115.

Show diffs side-by-side

added added

removed removed

Lines of Context:
34
34
           'get_exercise_attempts', 'get_exercise_attempt',
35
35
          ]
36
36
 
37
 
def get_exercise_status(store, user, worksheet_exercise, as_of=None):
 
37
def get_exercise_status(store, user, worksheet_exercise):
38
38
    """Given a storm.store, User and Exercise, returns information about
39
39
    the user's performance on that problem.
40
 
    @param store: A storm.store
41
 
    @param user: A User.
42
 
    @param worksheet_exercise: An Exercise.
43
 
    @param as_of: Optional datetime. If supplied, gets the status as of as_of.
44
40
    Returns a tuple of:
45
41
        - A boolean, whether they have successfully passed this exercise.
46
42
        - An int, the number of attempts they have made up to and
52
48
    is_relevant = ((ExerciseAttempt.user_id == user.id) &
53
49
            (ExerciseAttempt.ws_ex_id == worksheet_exercise.id) &
54
50
            (ExerciseAttempt.active == True))
55
 
    if as_of is not None:
56
 
        is_relevant &= ExerciseAttempt.date <= as_of
57
51
 
58
52
    # Get the first successful active attempt, or None if no success yet.
59
53
    # (For this user, for this exercise).
168
162
    saved.date = date
169
163
    saved.text = text
170
164
 
171
 
def calculate_score(store, user, worksheet, as_of=None):
 
165
def calculate_score(store, user, worksheet):
172
166
    """
173
167
    Given a storm.store, User, Exercise and Worksheet, calculates a score for
174
168
    the user on the given worksheet.
175
 
    @param store: A storm.store
176
 
    @param user: A User.
177
 
    @param worksheet: A Worksheet.
178
 
    @param as_of: Optional datetime. If supplied, gets the score as of as_of.
179
169
    Returns a 4-tuple of ints, consisting of:
180
170
    (No. mandatory exercises completed,
181
171
     Total no. mandatory exercises,
193
183
        worksheet = worksheet_exercise.worksheet
194
184
        optional = worksheet_exercise.optional
195
185
 
196
 
        done, _ = get_exercise_status(store, user, worksheet_exercise, as_of)
 
186
        done, _ = get_exercise_status(store, user, worksheet_exercise)
197
187
        # done is a bool, whether this student has completed that problem
198
188
        if optional:
199
189
            opt_total += 1
204
194
 
205
195
    return mand_done, mand_total, opt_done, opt_total
206
196
 
207
 
def calculate_mark(mand_done, mand_total):
208
 
    """Calculate a subject mark, given the result of all worksheets.
209
 
    @param mand_done: The total number of mandatory exercises completed by
210
 
        some student, across all worksheets.
211
 
    @param mand_total: The total number of mandatory exercises across all
212
 
        worksheets in the offering.
213
 
    @return: (percent, mark, mark_total)
214
 
        percent: The percentage of exercises the student has completed, as an
215
 
            integer between 0 and 100 inclusive.
216
 
        mark: The mark the student has received, based on the percentage.
217
 
        mark_total: The total number of marks available (currently hard-coded
218
 
            as 5).
219
 
    """
220
 
    # We want to display a students mark out of 5. However, they are
221
 
    # allowed to skip 1 in 5 questions and still get 'full marks'.
222
 
    # Hence we divide by 16, essentially making 16 percent worth
223
 
    # 1 star, and 80 or above worth 5.
224
 
    if mand_total > 0:
225
 
        percent_int = (100 * mand_done) // mand_total
226
 
    else:
227
 
        # Avoid Div0, just give everyone 0 marks if there are none available
228
 
        percent_int = 0
229
 
    # percent / 16, rounded down, with a maximum mark of 5
230
 
    max_mark = 5
231
 
    mark = min(percent_int // 16, max_mark)
232
 
    return (percent_int, mark, max_mark)
233
 
 
234
197
def update_exerciselist(worksheet):
235
198
    """Runs through the worksheetstream, generating the appropriate
236
199
    WorksheetExercises, and de-activating the old ones."""
237
200
    exercises = []
238
201
    # Turns the worksheet into an xml stream, and then finds all the 
239
202
    # exercise nodes in the stream.
240
 
    worksheetdata = genshi.XML(worksheet.get_xml())
 
203
    worksheetdata = genshi.XML(worksheet.data)
241
204
    for kind, data, pos in worksheetdata:
242
205
        if kind is genshi.core.START:
243
206
            # Data is a tuple of tag name and a list of name->value tuples