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

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

Undo r7102 and r7103.

  • Property svn:keywords set to Id
File size: 8.2 KB
RevLine 
[6621]1## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
2## This program is free software; you can redistribute it and/or modify
3## it under the terms of the GNU General Public License as published by
4## the Free Software Foundation; either version 2 of the License, or
5## (at your option) any later version.
6##
7## This program is distributed in the hope that it will be useful,
8## but WITHOUT ANY WARRANTY; without even the implied warranty of
9## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10## GNU General Public License for more details.
11##
12## You should have received a copy of the GNU General Public License
13## along with this program; if not, write to the Free Software
14## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15##
16"""
17Container for the various objects owned by students.
18"""
[7097]19import os
[6621]20import grok
21from grok import index
[6749]22from zope.component.interfaces import IFactory
[6621]23from zope.interface import implementedBy
[6838]24from hurry.workflow.interfaces import IWorkflowState, IWorkflowInfo
25from zope.securitypolicy.interfaces import IPrincipalRoleManager
[7097]26from waeup.sirp.interfaces import (
27    IObjectHistory, IUserAccount, IFileStoreNameChooser, IFileStoreHandler)
28from waeup.sirp.image import WAeUPImageFile
29from waeup.sirp.imagestorage import DefaultFileStoreHandler
[6694]30from waeup.sirp.students.interfaces import (
[6764]31    IStudent, IStudentNavigation, IStudentClearanceEdit,
[6695]32    IStudentPersonalEdit)
[6838]33from waeup.sirp.students.studycourse import StudentStudyCourse
[6859]34from waeup.sirp.students.payments import StudentPaymentsContainer
[6838]35from waeup.sirp.students.accommodation import StudentAccommodation
[6836]36from waeup.sirp.utils.helpers import attrs_to_fields, get_current_principal
[6651]37from waeup.sirp.students.utils import generate_student_id
[6621]38
39class Student(grok.Container):
40    """This is a student container for the various objects
41    owned by students.
42    """
[6694]43    grok.implements(IStudent, IStudentNavigation,
[6767]44        IStudentPersonalEdit, IStudentClearanceEdit)
[6621]45    grok.provides(IStudent)
46
47    def __init__(self):
48        super(Student, self).__init__()
[6749]49        # The site doesn't exist in unit tests
[6652]50        try:
[6749]51            students = grok.getSite()['students']
52            self.student_id = generate_student_id(students,'?')
53        except TypeError:
[6666]54            self.student_id = u'Z654321'
[6699]55        self.password = None
[6621]56        return
57
[6637]58    def loggerInfo(self, ob_class, comment=None):
59        target = self.__name__
60        return grok.getSite()['students'].logger_info(ob_class,target,comment)
61
62    @property
63    def state(self):
64        state = IWorkflowState(self).getState()
65        return state
66
67    @property
68    def history(self):
69        history = IObjectHistory(self)
70        return history
71
[6642]72    def getStudent(self):
73        return self
74
[6814]75    @property
76    def certificate(self):
77        cert = getattr(self.get('studycourse', None), 'certificate', None)
78        return cert
79
[7062]80    @property
81    def current_session(self):
82        cert = getattr(self.get('studycourse', None), 'current_session', None)
83        return cert
84
[6621]85# Set all attributes of Student required in IStudent as field
86# properties. Doing this, we do not have to set initial attributes
87# ourselves and as a bonus we get free validation when an attribute is
88# set.
89Student = attrs_to_fields(Student)
90
91class StudentFactory(grok.GlobalUtility):
92    """A factory for students.
93    """
94    grok.implements(IFactory)
95    grok.name(u'waeup.Student')
96    title = u"Create a new student.",
97    description = u"This factory instantiates new student instances."
98
99    def __call__(self, *args, **kw):
100        return Student()
101
102    def getInterfaces(self):
103        return implementedBy(Student)
[6836]104
[6838]105@grok.subscribe(IStudent, grok.IObjectAddedEvent)
[6839]106def handle_student_added(student, event):
[6838]107    """If a student is added all subcontainers are automatically added
108    and the transition create is fired. The latter produces a logging message.
109    """
110    student.clearance_locked = True
111    studycourse = StudentStudyCourse()
112    student['studycourse'] = studycourse
[6859]113    payments = StudentPaymentsContainer()
[6838]114    student['payments'] = payments
115    accommodation = StudentAccommodation()
116    student['accommodation'] = accommodation
117    # Assign global student role for new student
118    account = IUserAccount(student)
119    account.roles = ['waeup.Student']
120    # Assign local StudentRecordOwner role
121    role_manager = IPrincipalRoleManager(student)
122    role_manager.assignRoleToPrincipal(
123        'waeup.local.StudentRecordOwner', student.student_id)
124    IWorkflowInfo(student).fireTransition('create')
125    return
126
[6836]127@grok.subscribe(IStudent, grok.IObjectRemovedEvent)
[6839]128def handle_student_removed(student, event):
[6836]129    """If a student is removed a message is logged.
130    """
131    comment = 'Student record removed'
132    target = student.student_id
[6841]133    # In some tests we don't have a principal
134    try:
135        user = get_current_principal().id
136    except (TypeError, AttributeError):
137        return
[6836]138    grok.getSite()['students'].logger.info('%s - %s - %s' % (
[6838]139        user, target, comment))
[7097]140    return
141
142#: The file id marker for student files
143STUDENT_FILE_STORE_NAME = 'file-student'
144
145class StudentFileNameChooser(grok.Adapter):
[7099]146    """A file id chooser for :class:`Student` objects.
[7097]147
[7099]148    `context` is an :class:`Student` instance.
[7097]149
[7099]150    The :class:`StudentImageNameChooser` can build/check file ids for
151    :class:`Student` objects suitable for use with
[7097]152    :class:`ExtFileStore` instances. The delivered file_id contains
[7099]153    the file id marker for :class:`Student` object and the student id
154    of the context student.
[7097]155
156    This chooser is registered as an adapter providing
157    :class:`waeup.sirp.interfaces.IFileStoreNameChooser`.
158
159    File store name choosers like this one are only convenience
[7099]160    components to ease the task of creating file ids for student
[7097]161    objects. You are nevertheless encouraged to use them instead of
[7099]162    manually setting up filenames for students.
[7097]163
164    .. seealso:: :mod:`waeup.sirp.imagestorage`
165
166    """
167    grok.context(IStudent)
168    grok.implements(IFileStoreNameChooser)
169
170    def checkName(self, name=None, attr=None):
171        """Check whether the given name is a valid file id for the context.
172
173        Returns ``True`` only if `name` equals the result of
174        :meth:`chooseName`.
175
176        """
177        return name == self.chooseName()
178
[7105]179    def chooseName(self, name=None, attr=None):
[7097]180        """Get a valid file id for student context.
181
182        *Example:*
183
[7105]184        For a student with student id ``'A123456'`` and
185        with attr ``'nice_image'`` stored in
[7097]186        the students container this chooser would create:
187
188          ``'__file-student__students/A/A123456/nice_image_A123456.jpg'``
189
190        meaning that the nice image of this applicant would be
191        stored in the site-wide file storage in path:
192
193          ``students/A/A123456/nice_image_A123456.jpg``
194
195        """
[7099]196        stud_id = self.context.student_id
[7105]197        marked_filename = '__%s__%s/%s/%s_%s.jpg' % (
198            STUDENT_FILE_STORE_NAME, stud_id[0], stud_id, attr, stud_id)
[7097]199        return marked_filename
200
201
202class StudentFileStoreHandler(DefaultFileStoreHandler, grok.GlobalUtility):
203    """Student specific file handling.
204
205    This handler knows in which path in a filestore to store student
206    files and how to turn this kind of data into some (browsable)
207    file object.
208
209    It is called from the global file storage, when it wants to
210    get/store a file with a file id starting with
211    ``__file-student__`` (the marker string for student files).
212
213    Like each other file store handler it does not handle the files
214    really (this is done by the global file store) but only computes
215    paths and things like this.
216    """
217    grok.implements(IFileStoreHandler)
218    grok.name(STUDENT_FILE_STORE_NAME)
219
220    def pathFromFileID(self, store, root, file_id):
[7099]221        """All student files are put in directory ``students``.
[7097]222        """
223        marker, filename, basename, ext = store.extractMarker(file_id)
224        return os.path.join(root, 'students', filename)
225
226    def createFile(self, store, root, filename, file_id, file):
227        """Create a browsable file-like object.
228        """
229        # possible other actions: check for jpeg format
230        path = self.pathFromFileID(store, root, file_id)
231        return file, path, WAeUPImageFile(filename, file_id)
Note: See TracBrowser for help on using the repository browser.