source: main/waeup.sirp/trunk/src/waeup/sirp/students/student.py @ 7520

Last change on this file since 7520 was 7513, checked in by Henrik Bettermann, 13 years ago

Enable import of student registration states which are in IMPORTABLE_STATES.

Set workflow state directly rather than firing transitions.

  • Property svn:keywords set to Id
File size: 9.7 KB
Line 
1## $Id: student.py 7513 2012-01-25 17:47:23Z henrik $
2##
3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
18"""
19Container for the various objects owned by students.
20"""
21import os
22import grok
23from zope.component import getUtility
24from zope.component.interfaces import IFactory
25from zope.interface import implementedBy
26from hurry.workflow.interfaces import IWorkflowState, IWorkflowInfo
27from zope.securitypolicy.interfaces import IPrincipalRoleManager
28from waeup.sirp.interfaces import (
29    IObjectHistory, IUserAccount, IFileStoreNameChooser, IFileStoreHandler,
30    ISIRPUtils)
31from waeup.sirp.image import SIRPImageFile
32from waeup.sirp.imagestorage import DefaultFileStoreHandler
33from waeup.sirp.students.interfaces import (
34    IStudent, IStudentNavigation, IStudentClearanceEdit,
35    IStudentPersonalEdit)
36from waeup.sirp.students.studycourse import StudentStudyCourse
37from waeup.sirp.students.payments import StudentPaymentsContainer
38from waeup.sirp.students.accommodation import StudentAccommodation
39from waeup.sirp.utils.helpers import attrs_to_fields, get_current_principal
40from waeup.sirp.students.utils import generate_student_id
41
42class Student(grok.Container):
43    """This is a student container for the various objects
44    owned by students.
45    """
46    grok.implements(IStudent, IStudentNavigation,
47        IStudentPersonalEdit, IStudentClearanceEdit)
48    grok.provides(IStudent)
49
50    def __init__(self):
51        super(Student, self).__init__()
52        # The site doesn't exist in unit tests
53        try:
54            students = grok.getSite()['students']
55            self.student_id = generate_student_id(students,'?')
56        except TypeError:
57            self.student_id = u'Z654321'
58        self.password = None
59        return
60
61    def loggerInfo(self, ob_class, comment=None):
62        target = self.__name__
63        return grok.getSite()['students'].logger_info(ob_class,target,comment)
64
65    @property
66    def display_fullname(self):
67        middlename = getattr(self, 'middlename', None)
68        sirp_utils = getUtility(ISIRPUtils)
69        return sirp_utils.fullname(self.firstname, self.lastname, middlename)
70
71    @property
72    def fullname(self):
73        middlename = getattr(self, 'middlename', None)
74        if middlename:
75            return '%s-%s-%s' % (self.firstname.lower(),
76                middlename.lower(), self.lastname.lower())
77        else:
78            return '%s-%s' % (self.firstname.lower(), self.lastname.lower())
79
80    @property
81    def state(self):
82        state = IWorkflowState(self).getState()
83        return state
84
85    @property
86    def history(self):
87        history = IObjectHistory(self)
88        return history
89
90    def getStudent(self):
91        return self
92
93    @property
94    def certcode(self):
95        cert = getattr(self.get('studycourse', None), 'certificate', None)
96        if cert is not None:
97            return cert.code
98        return
99
100    @property
101    def faccode(self):
102        cert = getattr(self.get('studycourse', None), 'certificate', None)
103        if cert is not None:
104            return cert.__parent__.__parent__.__parent__.code
105        return
106
107    @property
108    def depcode(self):
109        cert = getattr(self.get('studycourse', None), 'certificate', None)
110        if cert is not None:
111            return cert.__parent__.__parent__.code
112        return
113
114    @property
115    def current_session(self):
116        cert = getattr(self.get('studycourse', None), 'current_session', None)
117        return cert
118
119# Set all attributes of Student required in IStudent as field
120# properties. Doing this, we do not have to set initial attributes
121# ourselves and as a bonus we get free validation when an attribute is
122# set.
123Student = attrs_to_fields(Student)
124
125class StudentFactory(grok.GlobalUtility):
126    """A factory for students.
127    """
128    grok.implements(IFactory)
129    grok.name(u'waeup.Student')
130    title = u"Create a new student.",
131    description = u"This factory instantiates new student instances."
132
133    def __call__(self, *args, **kw):
134        return Student()
135
136    def getInterfaces(self):
137        return implementedBy(Student)
138
139@grok.subscribe(IStudent, grok.IObjectAddedEvent)
140def handle_student_added(student, event):
141    """If a student is added all subcontainers are automatically added
142    and the transition create is fired. The latter produces a logging message.
143    """
144    student.clearance_locked = True
145    studycourse = StudentStudyCourse()
146    student['studycourse'] = studycourse
147    payments = StudentPaymentsContainer()
148    student['payments'] = payments
149    accommodation = StudentAccommodation()
150    student['accommodation'] = accommodation
151    # Assign global student role for new student
152    account = IUserAccount(student)
153    account.roles = ['waeup.Student']
154    # Assign local StudentRecordOwner role
155    role_manager = IPrincipalRoleManager(student)
156    role_manager.assignRoleToPrincipal(
157        'waeup.local.StudentRecordOwner', student.student_id)
158    if IWorkflowState(student).getState() is None:
159        IWorkflowInfo(student).fireTransition('create')
160    return
161
162@grok.subscribe(IStudent, grok.IObjectRemovedEvent)
163def handle_student_removed(student, event):
164    """If a student is removed a message is logged.
165    """
166    comment = 'Student record removed'
167    target = student.student_id
168    # In some tests we don't have a principal
169    try:
170        user = get_current_principal().id
171    except (TypeError, AttributeError):
172        return
173    try:
174        grok.getSite()['students'].logger.info('%s - %s - %s' % (
175            user, target, comment))
176    except KeyError:
177        # If we delete an entire university instance there won't be
178        # a students subcontainer
179        return
180    return
181
182#: The file id marker for student files
183STUDENT_FILE_STORE_NAME = 'file-student'
184
185class StudentFileNameChooser(grok.Adapter):
186    """A file id chooser for :class:`Student` objects.
187
188    `context` is an :class:`Student` instance.
189
190    The :class:`StudentImageNameChooser` can build/check file ids for
191    :class:`Student` objects suitable for use with
192    :class:`ExtFileStore` instances. The delivered file_id contains
193    the file id marker for :class:`Student` object and the student id
194    of the context student.
195
196    This chooser is registered as an adapter providing
197    :class:`waeup.sirp.interfaces.IFileStoreNameChooser`.
198
199    File store name choosers like this one are only convenience
200    components to ease the task of creating file ids for student
201    objects. You are nevertheless encouraged to use them instead of
202    manually setting up filenames for students.
203
204    .. seealso:: :mod:`waeup.sirp.imagestorage`
205
206    """
207    grok.context(IStudent)
208    grok.implements(IFileStoreNameChooser)
209
210    def checkName(self, name=None, attr=None):
211        """Check whether the given name is a valid file id for the context.
212
213        Returns ``True`` only if `name` equals the result of
214        :meth:`chooseName`.
215
216        """
217        return name == self.chooseName()
218
219    def chooseName(self, attr, name=None):
220        """Get a valid file id for student context.
221
222        *Example:*
223
224        For a student with student id ``'A123456'`` and
225        with attr ``'nice_image.jpeg'`` stored in
226        the students container this chooser would create:
227
228          ``'__file-student__students/A/A123456/nice_image_A123456.jpeg'``
229
230        meaning that the nice image of this applicant would be
231        stored in the site-wide file storage in path:
232
233          ``students/A/A123456/nice_image_A123456.jpeg``
234
235        """
236        basename, ext = os.path.splitext(attr)
237        stud_id = self.context.student_id
238        marked_filename = '__%s__%s/%s/%s_%s%s' % (
239            STUDENT_FILE_STORE_NAME, stud_id[0], stud_id, basename, stud_id, ext)
240        return marked_filename
241
242
243class StudentFileStoreHandler(DefaultFileStoreHandler, grok.GlobalUtility):
244    """Student specific file handling.
245
246    This handler knows in which path in a filestore to store student
247    files and how to turn this kind of data into some (browsable)
248    file object.
249
250    It is called from the global file storage, when it wants to
251    get/store a file with a file id starting with
252    ``__file-student__`` (the marker string for student files).
253
254    Like each other file store handler it does not handle the files
255    really (this is done by the global file store) but only computes
256    paths and things like this.
257    """
258    grok.implements(IFileStoreHandler)
259    grok.name(STUDENT_FILE_STORE_NAME)
260
261    def pathFromFileID(self, store, root, file_id):
262        """All student files are put in directory ``students``.
263        """
264        marker, filename, basename, ext = store.extractMarker(file_id)
265        sub_root = os.path.join(root, 'students')
266        return super(StudentFileStoreHandler, self).pathFromFileID(
267            store, sub_root, basename)
268
269    def createFile(self, store, root, filename, file_id, file):
270        """Create a browsable file-like object.
271        """
272        # call super method to ensure that any old files with
273        # different filename extension are deleted.
274        file, path, file_obj =  super(
275            StudentFileStoreHandler, self).createFile(
276            store, root,  filename, file_id, file)
277        return file, path, SIRPImageFile(
278            file_obj.filename, file_obj.data)
Note: See TracBrowser for help on using the repository browser.