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

1080.1.2 by matt.giuca
New module: ivle.database. Classes and utilities for Storm ORM.
1
# IVLE - Informatics Virtual Learning Environment
2
# Copyright (C) 2007-2009 The University of Melbourne
3
#
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
8
#
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
# GNU General Public License for more details.
13
#
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17
18
# Author: Matt Giuca, Will Grant
19
20
"""
21
Database Classes and Utilities for Storm ORM
22
23
This module provides all of the classes which map to database tables.
24
It also provides miscellaneous utility functions for database interaction.
25
"""
26
1080.1.13 by me at id
ivle.database.User: Add an authenticate() method, and a hash_password()
27
import md5
1080.1.15 by me at id
Give ivle.database.User {password,account}_expired attributes, and get
28
import datetime
1080.1.13 by me at id
ivle.database.User: Add an authenticate() method, and a hash_password()
29
1080.1.4 by matt.giuca
ivle.database: Added User class.
30
from storm.locals import create_database, Store, Int, Unicode, DateTime, \
1080.1.27 by me at id
ivle.database.User: Add an 'active_enrolments' property, which returns a list
31
                         Reference, ReferenceSet, Bool, Storm, Desc
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
32
from storm.exceptions import NotOneError, IntegrityError
1080.1.2 by matt.giuca
New module: ivle.database. Classes and utilities for Storm ORM.
33
34
import ivle.conf
1099.1.220 by Nick Chadwick
Merged from trunk
35
from ivle.worksheet.rst import rst
1080.1.2 by matt.giuca
New module: ivle.database. Classes and utilities for Storm ORM.
36
1080.1.39 by Matt Giuca
ivle.database: Added __all__ to the top of the file.
37
__all__ = ['get_store',
38
            'User',
39
            'Subject', 'Semester', 'Offering', 'Enrolment',
40
            'ProjectSet', 'Project', 'ProjectGroup', 'ProjectGroupMembership',
1080.1.59 by Matt Giuca
ivle.worksheet, ivle.database: Added/updated __all__.
41
            'Exercise', 'Worksheet', 'WorksheetExercise',
1080.1.61 by William Grant
ivle.database: Add an Offering.enrol(user) method, which enrols the user in
42
            'ExerciseSave', 'ExerciseAttempt',
1110 by William Grant
ivle-enrol now allows updating of existing enrolments. It also sets the role.
43
            'TestCase', 'TestSuite', 'TestSuiteVar'
1080.1.39 by Matt Giuca
ivle.database: Added __all__ to the top of the file.
44
        ]
45
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
46
def _kwarg_init(self, **kwargs):
47
    for k,v in kwargs.items():
1080.1.46 by William Grant
ivle.database._kwarg_init: Check with hasattr() on the class, not the object,
48
        if k.startswith('_') or not hasattr(self.__class__, k):
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
49
            raise TypeError("%s got an unexpected keyword argument '%s'"
1080.1.45 by William Grant
ivle.database._kwarg_init: Fix exception throwing.
50
                % (self.__class__.__name__, k))
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
51
        setattr(self, k, v)
52
1080.1.2 by matt.giuca
New module: ivle.database. Classes and utilities for Storm ORM.
53
def get_conn_string():
54
    """
55
    Returns the Storm connection string, generated from the conf file.
56
    """
1099.1.174 by William Grant
ivle.database.get_conn_string() now defaults to localhost:5432, rather than
57
58
    clusterstr = ''
59
    if ivle.conf.db_user:
60
        clusterstr += ivle.conf.db_user
61
        if ivle.conf.db_password:
62
            clusterstr += ':' + ivle.conf.db_password
63
        clusterstr += '@'
64
65
    host = ivle.conf.db_host or 'localhost'
66
    port = ivle.conf.db_port or 5432
67
68
    clusterstr += '%s:%d' % (host, port)
69
70
    return "postgres://%s/%s" % (clusterstr, ivle.conf.db_dbname)
1080.1.2 by matt.giuca
New module: ivle.database. Classes and utilities for Storm ORM.
71
72
def get_store():
73
    """
74
    Open a database connection and transaction. Return a storm.store.Store
75
    instance connected to the configured IVLE database.
76
    """
77
    return Store(create_database(get_conn_string()))
1080.1.4 by matt.giuca
ivle.database: Added User class.
78
1080.1.39 by Matt Giuca
ivle.database: Added __all__ to the top of the file.
79
# USERS #
80
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
81
class User(Storm):
1080.1.4 by matt.giuca
ivle.database: Added User class.
82
    """
83
    Represents an IVLE user.
84
    """
85
    __storm_table__ = "login"
86
87
    id = Int(primary=True, name="loginid")
88
    login = Unicode()
89
    passhash = Unicode()
90
    state = Unicode()
1101 by William Grant
Privileges (apart from admin) are now offering-local, not global.
91
    admin = Bool()
1080.1.4 by matt.giuca
ivle.database: Added User class.
92
    unixid = Int()
93
    nick = Unicode()
94
    pass_exp = DateTime()
95
    acct_exp = DateTime()
96
    last_login = DateTime()
97
    svn_pass = Unicode()
98
    email = Unicode()
99
    fullname = Unicode()
100
    studentid = Unicode()
101
    settings = Unicode()
102
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
103
    __init__ = _kwarg_init
1080.1.4 by matt.giuca
ivle.database: Added User class.
104
105
    def __repr__(self):
106
        return "<%s '%s'>" % (type(self).__name__, self.login)
1080.1.5 by matt.giuca
ivle.database.User: Add the missing methods from ivle.user.User.
107
1080.1.13 by me at id
ivle.database.User: Add an authenticate() method, and a hash_password()
108
    def authenticate(self, password):
109
        """Validate a given password against this user.
110
111
        Returns True if the given password matches the password hash for this
112
        User, False if it doesn't match, and None if there is no hash for the
113
        user.
114
        """
115
        if self.passhash is None:
116
            return None
117
        return self.hash_password(password) == self.passhash
118
1080.1.15 by me at id
Give ivle.database.User {password,account}_expired attributes, and get
119
    @property
120
    def password_expired(self):
1080.1.5 by matt.giuca
ivle.database.User: Add the missing methods from ivle.user.User.
121
        fieldval = self.pass_exp
1080.1.15 by me at id
Give ivle.database.User {password,account}_expired attributes, and get
122
        return fieldval is not None and datetime.datetime.now() > fieldval
123
124
    @property
125
    def account_expired(self):
1080.1.5 by matt.giuca
ivle.database.User: Add the missing methods from ivle.user.User.
126
        fieldval = self.acct_exp
1080.1.15 by me at id
Give ivle.database.User {password,account}_expired attributes, and get
127
        return fieldval is not None and datetime.datetime.now() > fieldval
1080.1.6 by matt.giuca
ivle.database.User: Added get_by_login method.
128
1099.1.121 by William Grant
Don't set req.user unless the login in the session specifies a valid user.
129
    @property
130
    def valid(self):
131
        return self.state == 'enabled' and not self.account_expired
132
1080.1.29 by me at id
ivle.database.User: Order 'enrolments' the same way as 'active_enrolments'.
133
    def _get_enrolments(self, justactive):
1080.1.27 by me at id
ivle.database.User: Add an 'active_enrolments' property, which returns a list
134
        return Store.of(self).find(Enrolment,
135
            Enrolment.user_id == self.id,
1080.1.29 by me at id
ivle.database.User: Order 'enrolments' the same way as 'active_enrolments'.
136
            (Enrolment.active == True) if justactive else True,
1080.1.27 by me at id
ivle.database.User: Add an 'active_enrolments' property, which returns a list
137
            Enrolment.offering_id == Offering.id,
138
            Offering.semester_id == Semester.id,
139
            Offering.subject_id == Subject.id).order_by(
140
                Desc(Semester.year),
141
                Desc(Semester.semester),
142
                Desc(Subject.code)
143
            )
144
1080.1.68 by William Grant
ivle.database.User: Add a write-only 'password' attribute. When set, it will
145
    def _set_password(self, password):
146
        if password is None:
147
            self.passhash = None
148
        else:
149
            self.passhash = unicode(User.hash_password(password))
150
    password = property(fset=_set_password)
151
1080.1.29 by me at id
ivle.database.User: Order 'enrolments' the same way as 'active_enrolments'.
152
    @property
1080.1.31 by me at id
ivle.database.User: Add 'subjects', an attribute containing currently
153
    def subjects(self):
154
        return Store.of(self).find(Subject,
155
            Enrolment.user_id == self.id,
156
            Enrolment.active == True,
157
            Offering.id == Enrolment.offering_id,
158
            Subject.id == Offering.subject_id).config(distinct=True)
159
1080.1.36 by William Grant
ivle.database: Add ProjectSet, Project, ProjectGroup, ProjectGroupMembership
160
    # TODO: Invitations should be listed too?
161
    def get_groups(self, offering=None):
162
        preds = [
163
            ProjectGroupMembership.user_id == self.id,
164
            ProjectGroup.id == ProjectGroupMembership.project_group_id,
165
        ]
166
        if offering:
167
            preds.extend([
168
                ProjectSet.offering_id == offering.id,
169
                ProjectGroup.project_set_id == ProjectSet.id,
170
            ])
171
        return Store.of(self).find(ProjectGroup, *preds)
172
173
    @property
174
    def groups(self):
175
        return self.get_groups()
176
1080.1.31 by me at id
ivle.database.User: Add 'subjects', an attribute containing currently
177
    @property
1080.1.29 by me at id
ivle.database.User: Order 'enrolments' the same way as 'active_enrolments'.
178
    def active_enrolments(self):
179
        '''A sanely ordered list of the user's active enrolments.'''
180
        return self._get_enrolments(True)
181
182
    @property
183
    def enrolments(self):
184
        '''A sanely ordered list of all of the user's enrolments.'''
185
        return self._get_enrolments(False) 
1080.1.27 by me at id
ivle.database.User: Add an 'active_enrolments' property, which returns a list
186
1080.1.13 by me at id
ivle.database.User: Add an authenticate() method, and a hash_password()
187
    @staticmethod
188
    def hash_password(password):
189
        return md5.md5(password).hexdigest()
190
1080.1.6 by matt.giuca
ivle.database.User: Added get_by_login method.
191
    @classmethod
192
    def get_by_login(cls, store, login):
193
        """
194
        Get the User from the db associated with a given store and
195
        login.
196
        """
1080.1.7 by matt.giuca
The new ivle.database.User class is now used in Request and usrmgt, which
197
        return store.find(cls, cls.login == unicode(login)).one()
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
198
1099.1.110 by William Grant
Implement an authorization system in the new framework. This breaks the REST
199
    def get_permissions(self, user):
1101 by William Grant
Privileges (apart from admin) are now offering-local, not global.
200
        if user and user.admin or user is self:
1099.1.110 by William Grant
Implement an authorization system in the new framework. This breaks the REST
201
            return set(['view', 'edit'])
202
        else:
203
            return set()
204
1080.1.39 by Matt Giuca
ivle.database: Added __all__ to the top of the file.
205
# SUBJECTS AND ENROLMENTS #
206
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
207
class Subject(Storm):
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
208
    __storm_table__ = "subject"
209
210
    id = Int(primary=True, name="subjectid")
211
    code = Unicode(name="subj_code")
212
    name = Unicode(name="subj_name")
213
    short_name = Unicode(name="subj_short_name")
214
    url = Unicode()
215
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
216
    offerings = ReferenceSet(id, 'Offering.subject_id')
217
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
218
    __init__ = _kwarg_init
219
220
    def __repr__(self):
221
        return "<%s '%s'>" % (type(self).__name__, self.short_name)
222
1099.1.110 by William Grant
Implement an authorization system in the new framework. This breaks the REST
223
    def get_permissions(self, user):
224
        perms = set()
225
        if user is not None:
226
            perms.add('view')
1101 by William Grant
Privileges (apart from admin) are now offering-local, not global.
227
            if user.admin:
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
228
                perms.add('edit')
1099.1.110 by William Grant
Implement an authorization system in the new framework. This breaks the REST
229
        return perms
230
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
231
class Semester(Storm):
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
232
    __storm_table__ = "semester"
233
234
    id = Int(primary=True, name="semesterid")
235
    year = Unicode()
236
    semester = Unicode()
1104 by William Grant
Replace Semester.active with Semester.state, allowing more useful state
237
    state = Unicode()
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
238
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
239
    offerings = ReferenceSet(id, 'Offering.semester_id')
1124 by William Grant
Add Semester.enrolments.
240
    enrolments = ReferenceSet(id,
241
                              'Offering.semester_id',
242
                              'Offering.id',
243
                              'Enrolment.offering_id')
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
244
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
245
    __init__ = _kwarg_init
246
247
    def __repr__(self):
248
        return "<%s %s/%s>" % (type(self).__name__, self.year, self.semester)
249
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
250
class Offering(Storm):
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
251
    __storm_table__ = "offering"
252
253
    id = Int(primary=True, name="offeringid")
254
    subject_id = Int(name="subject")
255
    subject = Reference(subject_id, Subject.id)
256
    semester_id = Int(name="semesterid")
257
    semester = Reference(semester_id, Semester.id)
258
    groups_student_permissions = Unicode()
259
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
260
    enrolments = ReferenceSet(id, 'Enrolment.offering_id')
1080.1.79 by William Grant
ivle.database.Offering: Add a members ReferenceSet.
261
    members = ReferenceSet(id,
262
                           'Enrolment.offering_id',
263
                           'Enrolment.user_id',
264
                           'User.id')
1080.1.76 by William Grant
ivle.database.Offering: Add project_sets referenceset.
265
    project_sets = ReferenceSet(id, 'ProjectSet.offering_id')
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
266
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
267
    worksheets = ReferenceSet(id, 
268
        'Worksheet.offering_id', 
1099.1.212 by Nick Chadwick
Added a new page to display exercises. This will then be modified to
269
        order_by="seq_no"
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
270
    )
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
271
1080.1.25 by me at id
ivle.database: Add Subject, Semester and Offering.
272
    __init__ = _kwarg_init
273
274
    def __repr__(self):
275
        return "<%s %r in %r>" % (type(self).__name__, self.subject,
276
                                  self.semester)
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
277
1110 by William Grant
ivle-enrol now allows updating of existing enrolments. It also sets the role.
278
    def enrol(self, user, role=u'student'):
1080.1.61 by William Grant
ivle.database: Add an Offering.enrol(user) method, which enrols the user in
279
        '''Enrol a user in this offering.'''
1110 by William Grant
ivle-enrol now allows updating of existing enrolments. It also sets the role.
280
        enrolment = Store.of(self).find(Enrolment,
1080.1.61 by William Grant
ivle.database: Add an Offering.enrol(user) method, which enrols the user in
281
                               Enrolment.user_id == user.id,
1110 by William Grant
ivle-enrol now allows updating of existing enrolments. It also sets the role.
282
                               Enrolment.offering_id == self.id).one()
283
284
        if enrolment is None:
285
            enrolment = Enrolment(user=user, offering=self)
286
            self.enrolments.add(enrolment)
287
288
        enrolment.active = True
289
        enrolment.role = role
1080.1.61 by William Grant
ivle.database: Add an Offering.enrol(user) method, which enrols the user in
290
1132 by William Grant
Add Offering.unenrol(), to unenrol a user from an offering.
291
    def unenrol(self, user):
292
        '''Unenrol a user from this offering.'''
293
        enrolment = Store.of(self).find(Enrolment,
294
                               Enrolment.user_id == user.id,
295
                               Enrolment.offering_id == self.id).one()
296
        Store.of(enrolment).remove(enrolment)
297
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
298
    def get_permissions(self, user):
299
        perms = set()
300
        if user is not None:
1131 by William Grant
Offerings now give 'view' only to user enrolled in them. 'edit' is granted
301
            enrolment = self.get_enrolment(user)
302
            if enrolment or user.admin:
303
                perms.add('view')
304
            if (enrolment and enrolment.role in (u'tutor', u'lecturer')) \
305
               or user.admin:
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
306
                perms.add('edit')
307
        return perms
308
1129 by William Grant
Move the group admin view to per-offering.
309
    def get_enrolment(self, user):
310
        try:
311
            enrolment = self.enrolments.find(user=user).one()
312
        except NotOneError:
313
            enrolment = None
314
315
        return enrolment
316
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
317
class Enrolment(Storm):
318
    __storm_table__ = "enrolment"
319
    __storm_primary__ = "user_id", "offering_id"
320
321
    user_id = Int(name="loginid")
322
    user = Reference(user_id, User.id)
323
    offering_id = Int(name="offeringid")
324
    offering = Reference(offering_id, Offering.id)
1101 by William Grant
Privileges (apart from admin) are now offering-local, not global.
325
    role = Unicode()
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
326
    notes = Unicode()
327
    active = Bool()
328
1080.1.81 by William Grant
ivle.database.Enrolment: Add a groups attribute, containing groups of which
329
    @property
330
    def groups(self):
331
        return Store.of(self).find(ProjectGroup,
332
                ProjectSet.offering_id == self.offering.id,
333
                ProjectGroup.project_set_id == ProjectSet.id,
334
                ProjectGroupMembership.project_group_id == ProjectGroup.id,
335
                ProjectGroupMembership.user_id == self.user.id)
336
1080.1.26 by me at id
ivle.database: Add an Enrolment class, and reference(set)s between all of the
337
    __init__ = _kwarg_init
338
339
    def __repr__(self):
340
        return "<%s %r in %r>" % (type(self).__name__, self.user,
341
                                  self.offering)
1080.1.36 by William Grant
ivle.database: Add ProjectSet, Project, ProjectGroup, ProjectGroupMembership
342
1080.1.39 by Matt Giuca
ivle.database: Added __all__ to the top of the file.
343
# PROJECTS #
344
1080.1.36 by William Grant
ivle.database: Add ProjectSet, Project, ProjectGroup, ProjectGroupMembership
345
class ProjectSet(Storm):
346
    __storm_table__ = "project_set"
347
348
    id = Int(name="projectsetid", primary=True)
349
    offering_id = Int(name="offeringid")
350
    offering = Reference(offering_id, Offering.id)
351
    max_students_per_group = Int()
352
1080.1.77 by William Grant
ivle.database.ProjectSet: Add projects and project_groups referencesets.
353
    projects = ReferenceSet(id, 'Project.project_set_id')
354
    project_groups = ReferenceSet(id, 'ProjectGroup.project_set_id')
355
1080.1.36 by William Grant
ivle.database: Add ProjectSet, Project, ProjectGroup, ProjectGroupMembership
356
    __init__ = _kwarg_init
357
358
    def __repr__(self):
359
        return "<%s %d in %r>" % (type(self).__name__, self.id,
360
                                  self.offering)
361
362
class Project(Storm):
363
    __storm_table__ = "project"
364
365
    id = Int(name="projectid", primary=True)
366
    synopsis = Unicode()
367
    url = Unicode()
368
    project_set_id = Int(name="projectsetid")
369
    project_set = Reference(project_set_id, ProjectSet.id)
370
    deadline = DateTime()
371
372
    __init__ = _kwarg_init
373
374
    def __repr__(self):
375
        return "<%s '%s' in %r>" % (type(self).__name__, self.synopsis,
376
                                  self.project_set.offering)
377
378
class ProjectGroup(Storm):
379
    __storm_table__ = "project_group"
380
381
    id = Int(name="groupid", primary=True)
382
    name = Unicode(name="groupnm")
383
    project_set_id = Int(name="projectsetid")
384
    project_set = Reference(project_set_id, ProjectSet.id)
385
    nick = Unicode()
386
    created_by_id = Int(name="createdby")
387
    created_by = Reference(created_by_id, User.id)
388
    epoch = DateTime()
389
1080.1.78 by William Grant
ivle.database.ProjectGroup.members: Use a ReferenceSet.
390
    members = ReferenceSet(id,
391
                           "ProjectGroupMembership.project_group_id",
392
                           "ProjectGroupMembership.user_id",
393
                           "User.id")
394
1080.1.36 by William Grant
ivle.database: Add ProjectSet, Project, ProjectGroup, ProjectGroupMembership
395
    __init__ = _kwarg_init
396
397
    def __repr__(self):
398
        return "<%s %s in %r>" % (type(self).__name__, self.name,
399
                                  self.project_set.offering)
400
401
class ProjectGroupMembership(Storm):
402
    __storm_table__ = "group_member"
403
    __storm_primary__ = "user_id", "project_group_id"
404
405
    user_id = Int(name="loginid")
406
    user = Reference(user_id, User.id)
407
    project_group_id = Int(name="groupid")
408
    project_group = Reference(project_group_id, ProjectGroup.id)
409
410
    __init__ = _kwarg_init
411
412
    def __repr__(self):
413
        return "<%s %r in %r>" % (type(self).__name__, self.user,
414
                                  self.project_group)
415
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
416
# WORKSHEETS AND EXERCISES #
417
418
class Exercise(Storm):
1099.1.195 by William Grant
Rename problem to exercise in the DB.
419
    __storm_table__ = "exercise"
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
420
    id = Unicode(primary=True, name="identifier")
421
    name = Unicode()
422
    description = Unicode()
423
    partial = Unicode()
424
    solution = Unicode()
425
    include = Unicode()
426
    num_rows = Int()
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
427
1099.6.2 by Nick Chadwick
Added a listing of all exercises
428
    worksheet_exercises =  ReferenceSet(id,
429
        'WorksheetExercise.exercise_id')
430
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
431
    worksheets = ReferenceSet(id,
432
        'WorksheetExercise.exercise_id',
433
        'WorksheetExercise.worksheet_id',
434
        'Worksheet.id'
435
    )
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
436
    
1099.1.212 by Nick Chadwick
Added a new page to display exercises. This will then be modified to
437
    test_suites = ReferenceSet(id, 
438
        'TestSuite.exercise_id',
439
        order_by='seq_no')
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
440
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
441
    __init__ = _kwarg_init
442
443
    def __repr__(self):
444
        return "<%s %s>" % (type(self).__name__, self.name)
445
1099.1.210 by Nick Chadwick
Modified the database layer so that exercises have a get_permissions
446
    def get_permissions(self, user):
447
        perms = set()
1099.1.234 by Nick Chadwick
Permissions for editing and deleting exercises now come from the
448
        roles = set()
1099.1.210 by Nick Chadwick
Modified the database layer so that exercises have a get_permissions
449
        if user is not None:
1101 by William Grant
Privileges (apart from admin) are now offering-local, not global.
450
            if user.admin:
1099.1.210 by Nick Chadwick
Modified the database layer so that exercises have a get_permissions
451
                perms.add('edit')
452
                perms.add('view')
1099.1.236 by Nick Chadwick
Fixed a syntax error.
453
            elif 'lecturer' in set((e.role for e in user.active_enrolments)):
1099.1.234 by Nick Chadwick
Permissions for editing and deleting exercises now come from the
454
                perms.add('edit')
455
                perms.add('view')
1099.1.235 by Nick Chadwick
Made checking if a user is a lecturer in exercise get_permissions
456
            
1099.1.210 by Nick Chadwick
Modified the database layer so that exercises have a get_permissions
457
        return perms
1099.6.3 by Nick Chadwick
Edited the exercise service to delete individual parts of an exercise.
458
    
459
    def get_description(self):
1099.1.232 by Nick Chadwick
Removed XML from database. RST now generates a full xml document, not
460
        return rst(self.description)
1080.1.51 by Matt Giuca
tutorial: Replaced call to ivle.db.create_worksheet with local code (roughly
461
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
462
    def delete(self):
463
        """Deletes the exercise, providing it has no associated worksheets."""
464
        if (self.worksheet_exercises.count() > 0):
465
            raise IntegrityError()
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
466
        for suite in self.test_suites:
467
            suite.delete()
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
468
        Store.of(self).remove(self)
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
469
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
470
class Worksheet(Storm):
471
    __storm_table__ = "worksheet"
472
473
    id = Int(primary=True, name="worksheetid")
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
474
    offering_id = Int(name="offeringid")
1099.4.1 by Nick Chadwick
Working on putting worksheets into the database.
475
    identifier = Unicode()
476
    name = Unicode()
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
477
    assessable = Bool()
1099.4.1 by Nick Chadwick
Working on putting worksheets into the database.
478
    data = Unicode()
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
479
    seq_no = Int()
480
    format = Unicode()
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
481
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
482
    attempts = ReferenceSet(id, "ExerciseAttempt.worksheetid")
1099.1.118 by William Grant
Fix a bad reference introduced with the worksheet changes.
483
    offering = Reference(offering_id, 'Offering.id')
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
484
1103 by William Grant
Worksheet.worksheet_exercises now only contains active ones.
485
    all_worksheet_exercises = ReferenceSet(id,
486
        'WorksheetExercise.worksheet_id')
487
488
    # Use worksheet_exercises to get access to the *active* WorksheetExercise
489
    # objects binding worksheets to exercises. This is required to access the
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
490
    # "optional" field.
1099.1.220 by Nick Chadwick
Merged from trunk
491
1103 by William Grant
Worksheet.worksheet_exercises now only contains active ones.
492
    @property
493
    def worksheet_exercises(self):
494
        return self.all_worksheet_exercises.find(active=True)
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
495
1080.1.40 by Matt Giuca
ivle.database: Added Worksheet and Exercise classes (more to come in this
496
    __init__ = _kwarg_init
497
498
    def __repr__(self):
499
        return "<%s %s>" % (type(self).__name__, self.name)
1080.1.47 by Matt Giuca
ivle.database: Added Worksheet.get_by_name method.
500
501
    # XXX Refactor this - make it an instance method of Subject rather than a
502
    # class method of Worksheet. Can't do that now because Subject isn't
503
    # linked referentially to the Worksheet.
504
    @classmethod
505
    def get_by_name(cls, store, subjectname, worksheetname):
506
        """
507
        Get the Worksheet from the db associated with a given store, subject
508
        name and worksheet name.
509
        """
510
        return store.find(cls, cls.subject == unicode(subjectname),
511
            cls.name == unicode(worksheetname)).one()
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
512
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
513
    def remove_all_exercises(self):
1080.1.51 by Matt Giuca
tutorial: Replaced call to ivle.db.create_worksheet with local code (roughly
514
        """
515
        Remove all exercises from this worksheet.
516
        This does not delete the exercises themselves. It just removes them
517
        from the worksheet.
518
        """
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
519
        store = Store.of(self)
520
        for ws_ex in self.all_worksheet_exercises:
521
            if ws_ex.saves.count() > 0 or ws_ex.attempts.count() > 0:
522
                raise IntegrityError()
1080.1.51 by Matt Giuca
tutorial: Replaced call to ivle.db.create_worksheet with local code (roughly
523
        store.find(WorksheetExercise,
524
            WorksheetExercise.worksheet == self).remove()
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
525
            
526
    def get_permissions(self, user):
527
        return self.offering.get_permissions(user)
1099.1.220 by Nick Chadwick
Merged from trunk
528
    
529
    def get_xml(self):
530
        """Returns the xml of this worksheet, converts from rst if required."""
531
        if self.format == u'rst':
1099.1.232 by Nick Chadwick
Removed XML from database. RST now generates a full xml document, not
532
            ws_xml = rst(self.data)
1099.1.220 by Nick Chadwick
Merged from trunk
533
            return ws_xml
534
        else:
535
            return self.data
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
536
    
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
537
    def delete(self):
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
538
        """Deletes the worksheet, provided it has no attempts on any exercises.
539
        
540
        Returns True if delete succeeded, or False if this worksheet has
541
        attempts attached."""
542
        for ws_ex in self.all_worksheet_exercises:
543
            if ws_ex.saves.count() > 0 or ws_ex.attempts.count() > 0:
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
544
                raise IntegrityError()
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
545
        
546
        self.remove_all_exercises()
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
547
        Store.of(self).remove(self)
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
548
        
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
549
class WorksheetExercise(Storm):
1099.1.195 by William Grant
Rename problem to exercise in the DB.
550
    __storm_table__ = "worksheet_exercise"
1099.4.4 by Nick Chadwick
Made what should (hopefully) be the last changes to the database schema.
551
    
1099.1.195 by William Grant
Rename problem to exercise in the DB.
552
    id = Int(primary=True, name="ws_ex_id")
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
553
554
    worksheet_id = Int(name="worksheetid")
555
    worksheet = Reference(worksheet_id, Worksheet.id)
1099.1.195 by William Grant
Rename problem to exercise in the DB.
556
    exercise_id = Unicode(name="exerciseid")
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
557
    exercise = Reference(exercise_id, Exercise.id)
558
    optional = Bool()
1099.4.3 by Nick Chadwick
Updated the tutorial service, to now allow users to edit worksheets
559
    active = Bool()
560
    seq_no = Int()
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
561
    
562
    saves = ReferenceSet(id, "ExerciseSave.ws_ex_id")
1099.1.183 by William Grant
Fix a reference typo in ivle.database.
563
    attempts = ReferenceSet(id, "ExerciseAttempt.ws_ex_id")
1080.1.50 by Matt Giuca
ivle.database: Added WorksheetExercise (relates worksheets to exercises), and
564
565
    __init__ = _kwarg_init
566
567
    def __repr__(self):
568
        return "<%s %s in %s>" % (type(self).__name__, self.exercise.name,
1099.4.1 by Nick Chadwick
Working on putting worksheets into the database.
569
                                  self.worksheet.identifier)
1080.1.55 by Matt Giuca
ivle.database: Added ExerciseAttempt and ExerciseSave classes.
570
1131 by William Grant
Offerings now give 'view' only to user enrolled in them. 'edit' is granted
571
    def get_permissions(self, user):
572
        return self.worksheet.get_permissions(user)
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
573
    
1131 by William Grant
Offerings now give 'view' only to user enrolled in them. 'edit' is granted
574
1080.1.55 by Matt Giuca
ivle.database: Added ExerciseAttempt and ExerciseSave classes.
575
class ExerciseSave(Storm):
576
    """
577
    Represents a potential solution to an exercise that a user has submitted
578
    to the server for storage.
579
    A basic ExerciseSave is just the current saved text for this exercise for
580
    this user (doesn't count towards their attempts).
581
    ExerciseSave may be extended with additional semantics (such as
582
    ExerciseAttempt).
583
    """
1099.1.195 by William Grant
Rename problem to exercise in the DB.
584
    __storm_table__ = "exercise_save"
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
585
    __storm_primary__ = "ws_ex_id", "user_id"
586
1099.1.195 by William Grant
Rename problem to exercise in the DB.
587
    ws_ex_id = Int(name="ws_ex_id")
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
588
    worksheet_exercise = Reference(ws_ex_id, "WorksheetExercise.id")
589
1080.1.55 by Matt Giuca
ivle.database: Added ExerciseAttempt and ExerciseSave classes.
590
    user_id = Int(name="loginid")
591
    user = Reference(user_id, User.id)
592
    date = DateTime()
593
    text = Unicode()
594
595
    __init__ = _kwarg_init
596
597
    def __repr__(self):
598
        return "<%s %s by %s at %s>" % (type(self).__name__,
599
            self.exercise.name, self.user.login, self.date.strftime("%c"))
600
601
class ExerciseAttempt(ExerciseSave):
602
    """
603
    An ExerciseAttempt is a special case of an ExerciseSave. Like an
604
    ExerciseSave, it constitutes exercise solution data that the user has
605
    submitted to the server for storage.
606
    In addition, it contains additional information about the submission.
607
    complete - True if this submission was successful, rendering this exercise
608
        complete for this user.
609
    active - True if this submission is "active" (usually true). Submissions
610
        may be de-activated by privileged users for special reasons, and then
611
        they won't count (either as a penalty or success), but will still be
612
        stored.
613
    """
1099.1.195 by William Grant
Rename problem to exercise in the DB.
614
    __storm_table__ = "exercise_attempt"
1099.1.180 by Nick Chadwick
This commit changes the tutorial service, which now almost exclusively
615
    __storm_primary__ = "ws_ex_id", "user_id", "date"
1080.1.55 by Matt Giuca
ivle.database: Added ExerciseAttempt and ExerciseSave classes.
616
617
    # The "text" field is the same but has a different name in the DB table
618
    # for some reason.
619
    text = Unicode(name="attempt")
620
    complete = Bool()
621
    active = Bool()
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
622
    
1099.1.113 by William Grant
Give console and tutorial services security declarations.
623
    def get_permissions(self, user):
624
        return set(['view']) if user is self.user else set()
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
625
  
626
class TestSuite(Storm):
627
    """A Testsuite acts as a container for the test cases of an exercise."""
628
    __storm_table__ = "test_suite"
629
    __storm_primary__ = "exercise_id", "suiteid"
630
    
631
    suiteid = Int()
1099.1.195 by William Grant
Rename problem to exercise in the DB.
632
    exercise_id = Unicode(name="exerciseid")
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
633
    description = Unicode()
634
    seq_no = Int()
635
    function = Unicode()
636
    stdin = Unicode()
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
637
    exercise = Reference(exercise_id, Exercise.id)
1099.1.212 by Nick Chadwick
Added a new page to display exercises. This will then be modified to
638
    test_cases = ReferenceSet(suiteid, 'TestCase.suiteid', order_by="seq_no")
639
    variables = ReferenceSet(suiteid, 'TestSuiteVar.suiteid', order_by='arg_no')
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
640
    
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
641
    def delete(self):
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
642
        """Delete this suite, without asking questions."""
643
        for vaariable in self.variables:
644
            variable.delete()
645
        for test_case in self.test_cases:
646
            test_case.delete()
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
647
        Store.of(self).remove(self)
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
648
649
class TestCase(Storm):
650
    """A TestCase is a member of a TestSuite.
651
    
652
    It contains the data necessary to check if an exercise is correct"""
653
    __storm_table__ = "test_case"
654
    __storm_primary__ = "testid", "suiteid"
655
    
656
    testid = Int()
657
    suiteid = Int()
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
658
    suite = Reference(suiteid, "TestSuite.suiteid")
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
659
    passmsg = Unicode()
660
    failmsg = Unicode()
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
661
    test_default = Unicode()
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
662
    seq_no = Int()
663
    
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
664
    parts = ReferenceSet(testid, "TestCasePart.testid")
665
    
666
    __init__ = _kwarg_init
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
667
    
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
668
    def delete(self):
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
669
        for part in self.parts:
670
            part.delete()
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
671
        Store.of(self).remove(self)
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
672
673
class TestSuiteVar(Storm):
674
    """A container for the arguments of a Test Suite"""
1099.1.195 by William Grant
Rename problem to exercise in the DB.
675
    __storm_table__ = "suite_variable"
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
676
    __storm_primary__ = "varid"
677
    
678
    varid = Int()
679
    suiteid = Int()
680
    var_name = Unicode()
681
    var_value = Unicode()
682
    var_type = Unicode()
683
    arg_no = Int()
684
    
685
    suite = Reference(suiteid, "TestSuite.suiteid")
686
    
687
    __init__ = _kwarg_init
688
    
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
689
    def delete(self):
690
        Store.of(self).remove(self)
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
691
    
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
692
class TestCasePart(Storm):
693
    """A container for the test elements of a Test Case"""
1099.1.195 by William Grant
Rename problem to exercise in the DB.
694
    __storm_table__ = "test_case_part"
1099.1.141 by Nick Chadwick
Updated the exercises to be loaded from the database, not a local file.
695
    __storm_primary__ = "partid"
696
    
697
    partid = Int()
698
    testid = Int()
699
    
700
    part_type = Unicode()
701
    test_type = Unicode()
702
    data = Unicode()
703
    filename = Unicode()
704
    
705
    test = Reference(testid, "TestCase.testid")
706
    
1099.1.114 by Nick Chadwick
Modified the database so that exercises are now stored in the database, rather
707
    __init__ = _kwarg_init
1099.1.233 by Nick Chadwick
Exercise objects in the database module, along with their test cases,
708
    
1099.1.242 by Nick Chadwick
Fixed a problem with exercise editor, which wasn't editing or adding
709
    def delete(self):
710
        Store.of(self).remove(self)