source: main/waeup.kofa/trunk/src/waeup/kofa/applicants/applicant.py @ 17072

Last change on this file since 17072 was 17055, checked in by Henrik Bettermann, 2 years ago

Doesn't work.

  • Property svn:keywords set to Id
File size: 20.1 KB
RevLine 
[7192]1## $Id: applicant.py 17055 2022-08-02 22:53:36Z henrik $
[5272]2##
[7192]3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
[5272]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.
[7192]8##
[5272]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.
[7192]13##
[5272]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##
[7063]18import os
[5272]19import grok
[7391]20from cStringIO import StringIO
[6115]21from grok import index
[8404]22from hurry.query import Eq, Text
23from hurry.query.query import Query
[16551]24from urllib import urlencode
[7438]25from zope.component import getUtility, createObject, getAdapter
[5318]26from zope.component.interfaces import IFactory
[7421]27from zope.event import notify
[7240]28from zope.securitypolicy.interfaces import IPrincipalRoleManager
[5479]29from zope.interface import implementedBy
[16278]30from zope.schema.interfaces import (
31    RequiredMissing, ConstraintNotSatisfied, WrongType)
[6296]32from hurry.workflow.interfaces import IWorkflowInfo, IWorkflowState
[7819]33from waeup.kofa.image import KofaImageFile
[14256]34from reportlab.platypus.doctemplate import LayoutError
[7811]35from waeup.kofa.imagestorage import DefaultFileStoreHandler
36from waeup.kofa.interfaces import (
[7819]37    IObjectHistory, IFileStoreHandler, IFileStoreNameChooser, IKofaUtils,
[9217]38    IExtFileStore, IPDF, IUserAccount, IUniversity)
[8311]39from waeup.kofa.interfaces import MessageFactory as _
[7811]40from waeup.kofa.students.vocabularies import RegNumNotInSource
[8843]41from waeup.kofa.students.studycourse import StudentStudyCourse
[7811]42from waeup.kofa.utils.helpers import attrs_to_fields
[10845]43from waeup.kofa.applicants.interfaces import (
[15941]44    IApplicant, IApplicantEdit, ISpecialApplicant, IApplicantsUtils)
[7811]45from waeup.kofa.applicants.workflow import application_states_dict
[13968]46from waeup.kofa.applicants.payment import ApplicantOnlinePayment
[13974]47from waeup.kofa.applicants.refereereport import ApplicantRefereeReport
[5272]48
[8404]49def search(query=None, searchtype=None, view=None):
50    if searchtype in ('fullname',):
51        results = Query().searchResults(
52            Text(('applicants_catalog', searchtype), query))
53    else:
54        results = Query().searchResults(
55            Eq(('applicants_catalog', searchtype), query))
56    return results
57
[7250]58class Applicant(grok.Container):
[10845]59    grok.implements(IApplicant, IApplicantEdit, ISpecialApplicant)
[5483]60    grok.provides(IApplicant)
[5272]61
[11615]62    applicant_student_mapping = [
[11614]63        ('firstname', 'firstname'),
64        ('middlename', 'middlename'),
65        ('lastname', 'lastname'),
66        ('sex', 'sex'),
67        ('date_of_birth', 'date_of_birth'),
68        ('email', 'email'),
69        ('phone', 'phone'),
[11615]70        ]
[8667]71
[16228]72    applicant_graduated_mapping = [
73        ('firstname', 'firstname'),
74        ('middlename', 'middlename'),
75        ('lastname', 'lastname'),
76        ('sex', 'sex'),
77        ('date_of_birth', 'date_of_birth'),
78        ('email', 'email'),
79        ('phone', 'phone'),
80        ]
81
[7260]82    def __init__(self):
[6296]83        super(Applicant, self).__init__()
[7240]84        self.password = None
[6476]85        self.application_date = None
[7260]86        self.applicant_id = None
[6331]87        return
[6296]88
[13968]89    @property
90    def payments(self):
91        payments = [value for value in self.values()
92            if isinstance(value, ApplicantOnlinePayment)]
93        return payments
94
[13974]95    @property
96    def refereereports(self):
97        reports = [value for value in self.values()
98            if isinstance(value, ApplicantRefereeReport)]
99        return reports
100
[8742]101    def writeLogMessage(self, view, message):
102        ob_class = view.__implemented__.__name__.replace('waeup.kofa.','')
103        self.__parent__.__parent__.logger.info(
104            '%s - %s - %s' % (ob_class, self.applicant_id, message))
105        return
[6475]106
[6324]107    @property
[8286]108    def state(self):
[7686]109        return IWorkflowState(self).getState()
[6324]110
[6339]111    @property
[8377]112    def container_code(self):
[13080]113        try:
114            code = self.__parent__.code
115        except AttributeError:  # in unit tests
116            return
[13216]117        if (self.password,
118            self.firstname,
119            self.lastname,
120            self.email) == (None, None, None, None):
121            return code + '-'
122        return code + '+'
[8377]123
124    @property
[7686]125    def translated_state(self):
[13080]126        try:
127            state = application_states_dict[self.state]
128        except LookupError:  # in unit tests
129            return
130        return state
[7686]131
132    @property
[6339]133    def history(self):
134        history = IObjectHistory(self)
135        return history
136
[7240]137    @property
[10831]138    def special(self):
[13080]139        try:
140            special = self.__parent__.prefix.startswith('special')
141        except AttributeError:  # in unit tests
142            return
143        return special
[10831]144
145    @property
[7240]146    def application_number(self):
[7260]147        try:
148            return self.applicant_id.split('_')[1]
149        except AttributeError:
150            return None
[7240]151
152    @property
[7364]153    def display_fullname(self):
[7356]154        middlename = getattr(self, 'middlename', None)
[7819]155        kofa_utils = getUtility(IKofaUtils)
[7811]156        return kofa_utils.fullname(self.firstname, self.lastname, middlename)
[7240]157
[9480]158    def _setStudyCourseAttributes(self, studycourse):
159        studycourse.entry_mode = self.course_admitted.study_mode
160        studycourse.current_level = self.course_admitted.start_level
161        studycourse.certificate = self.course_admitted
162        studycourse.entry_session = self.__parent__.year
163        studycourse.current_session = self.__parent__.year
164        return
165
[16228]166    def _setGraduatedStudyCourseAttributes(self, studycourse):
167        studycourse.entry_mode = self.course_studied.study_mode
168        studycourse.current_level = self.course_studied.start_level
169        studycourse.certificate = self.course_studied
170        studycourse.entry_session = self.__parent__.year
171        studycourse.current_session = self.__parent__.year
172        return
173
[16551]174    def createStudent(self, view=None, graduated=False, send_email=False):
[15941]175        """Create a student, fill with base data, create an application slip,
[16551]176        copy applicant data and files, send an email (optional).
[7338]177        """
[15932]178        site = grok.getSite()
[7344]179        # Is applicant in the correct state?
[16228]180        if graduated:
181            if self.state != 'processed':
182                return False, _('Applicant has not yet been processed.')
183            certificate = getattr(self, 'course_studied', None)
184            mapping = self.applicant_graduated_mapping
185        else:
186            if self.state != 'admitted':
187                return False, _('Applicant has not yet been admitted.')
188            certificate = getattr(self, 'course_admitted', None)
189            mapping = self.applicant_student_mapping
[16955]190        # Set (fallback) reg_number
191        if self.reg_number is None:
192            self.reg_number = self.applicant_id
[7344]193        # Does registration number exist?
[7338]194        student = createObject(u'waeup.Student')
[7341]195        try:
196            student.reg_number = self.reg_number
197        except RegNumNotInSource:
[15932]198            # Reset _curr_stud_id
199            site['students']._curr_stud_id -= 1
[8311]200            return False, _('Registration Number exists.')
[16228]201        # Has the course_admitted/course_studied field been properly filled?
202        if certificate is None:
[15932]203            # Reset _curr_stud_id
204            site['students']._curr_stud_id -= 1
[16228]205            return False, _('No study course provided.')
[8843]206        # Set student attributes
[8635]207        try:
[16228]208            for item in mapping:
[16279]209                if item[0] == 'email':
[16282]210                    # The field type in the application section might be
211                    # TextLineChoice to guarantee uniqueness of email addresses.
[16279]212                    setattr(student, item[1], str(getattr(self, item[0], None)))
213                else:
214                    setattr(student, item[1], getattr(self, item[0], None))
[8635]215        except RequiredMissing, err:
[15932]216            site['students']._curr_stud_id -= 1
[8855]217            return False, 'RequiredMissing: %s' % err
[16278]218        except WrongType, err:
[16279]219            site['students']._curr_stud_id -= 1
220            return False, 'WrongType: %s' % err
[16598]221        except UnicodeEncodeError, err:
222            return False, 'UnicodeEncodeError: %s' % err
[8843]223        except:
[15932]224            site['students']._curr_stud_id -= 1
[9391]225            return False, 'Unknown Error'
[12395]226        # Prove if the certificate still exists
[8843]227        try:
[16228]228            StudentStudyCourse().certificate = certificate
[8843]229        except ConstraintNotSatisfied, err:
[15932]230            # Reset _curr_stud_id
231            site['students']._curr_stud_id -= 1
[16228]232            return False, 'ConstraintNotSatisfied: %s' % certificate.code
[12395]233        # Finally prove if an application slip can be created
234        try:
[12565]235            test_applicant_slip = getAdapter(self, IPDF, name='application_slip')(
[12395]236                view=view)
237        except IOError:
[15932]238            site['students']._curr_stud_id -= 1
[12395]239            return False, _('IOError: Application Slip could not be created.')
[14256]240        except LayoutError, err:
[15932]241            site['students']._curr_stud_id -= 1
[14256]242            return False, _('Reportlab LayoutError: %s' % err)
[15930]243        except AttributeError, err:
[15932]244            site['students']._curr_stud_id -= 1
[15930]245            return False, _('Reportlab AttributeError: ${a}', mapping = {'a':err})
[14263]246        except:
[15932]247            site['students']._curr_stud_id -= 1
[15930]248            return False, _('Unknown Reportlab error')
[8843]249        # Add student
[7406]250        site['students'].addStudent(student)
[9121]251        # Save student_id
252        self.student_id = student.student_id
[17055]253        if graduated:
254            # Set state
255            IWorkflowState(student).setState('graduated')
256            # Save the certificate and set study course attributes
257            self._setGraduatedStudyCourseAttributes(student['studycourse'])
258        else:
259            # Fire transitions
260            IWorkflowInfo(self).fireTransition('create')
261            IWorkflowInfo(student).fireTransition('admit')
262            # Save the certificate and set study course attributes
263            self._setStudyCourseAttributes(student['studycourse'])
[7406]264        # Set password
[7409]265        IUserAccount(student).setPassword(self.application_number)
[7388]266        self._copyPassportImage(student)
[15941]267        self._copyFiles(student)
[7421]268        # Update the catalog
269        notify(grok.ObjectModifiedEvent(student))
[7391]270        # Save application slip
[12565]271        applicant_slip = getAdapter(self, IPDF, name='application_slip')(
272            view=view)
[12395]273        self._saveApplicationPDF(student, applicant_slip, view=view)
[16551]274        if view and send_email:
275            kofa_utils = getUtility(IKofaUtils)
276            pw = self.application_number
277            args = {'login':student.student_id, 'password':pw}
278            login_url = view.url(grok.getSite()) + '/login?%s' % urlencode(args)
279            rpw_url = view.url(grok.getSite(), 'changepw')
280            kofa_utils.informNewStudent(
281                IUserAccount(student), pw, login_url, rpw_url)
[12395]282        return True, _('Student ${a} created', mapping = {'a':student.student_id})
[7355]283
[12395]284    def _saveApplicationPDF(self, student, applicant_slip, view=None):
[7391]285        """Create an application slip as PDF and store it in student folder.
286        """
287        file_store = getUtility(IExtFileStore)
288        file_id = IFileStoreNameChooser(student).chooseName(
289            attr="application_slip.pdf")
290        file_store.createFile(file_id, StringIO(applicant_slip))
291        return
292
[7388]293    def _copyPassportImage(self, student):
294        """Copy any passport image over to student location.
295        """
296        file_store = getUtility(IExtFileStore)
297        appl_file = file_store.getFileByContext(self)
[7417]298        if appl_file is None:
299            return
[7388]300        stud_file_id = IFileStoreNameChooser(student).chooseName(
301            attr="passport.jpg")
[7417]302        file_store.createFile(stud_file_id, appl_file)
[7388]303        return
304
[15941]305    def _copyFiles(self, student):
306        """Copy all other files over to student location. Not
307        used in base package but tested with fake file.
308        """
309        file_store = getUtility(IExtFileStore)
[15943]310        filenames = getUtility(IApplicantsUtils).ADDITIONAL_FILES
[15941]311        for filename in filenames:
[15943]312            appl_file = file_store.getFileByContext(self, attr=filename[1])
[15941]313            if appl_file is None:
[16016]314                continue
[16545]315            ext = os.path.splitext(appl_file.name)[1]
[15941]316            stud_file_id = IFileStoreNameChooser(student).chooseName(
[16545]317                attr=filename[1] + ext)
[15941]318            file_store.createFile(stud_file_id, appl_file)
319        return
320
[5983]321# Set all attributes of Applicant required in IApplicant as field
322# properties. Doing this, we do not have to set initial attributes
323# ourselves and as a bonus we get free validation when an attribute is
324# set.
[6115]325Applicant = attrs_to_fields(Applicant)
[5272]326
[8404]327class ApplicantsCatalog(grok.Indexes):
[6115]328    """A catalog indexing :class:`Applicant` instances in the ZODB.
329    """
[9217]330    grok.site(IUniversity)
[6115]331    grok.name('applicants_catalog')
332    grok.context(IApplicant)
333
[8404]334    fullname = index.Text(attribute='display_fullname')
[7240]335    applicant_id = index.Field(attribute='applicant_id')
[7270]336    reg_number = index.Field(attribute='reg_number')
[8039]337    email = index.Field(attribute='email')
[8635]338    state = index.Field(attribute='state')
[13216]339    container_code = index.Field(attribute='container_code')
[6115]340
[5318]341class ApplicantFactory(grok.GlobalUtility):
[6620]342    """A factory for applicants.
[5318]343    """
344    grok.implements(IFactory)
345    grok.name(u'waeup.Applicant')
346    title = u"Create a new applicant.",
347    description = u"This factory instantiates new applicant instances."
348
349    def __call__(self, *args, **kw):
[7260]350        return Applicant()
[5318]351
352    def getInterfaces(self):
[5479]353        return implementedBy(Applicant)
[7063]354
355
356#: The file id marker for applicant passport images
357APPLICANT_IMAGE_STORE_NAME = 'img-applicant'
358
359class ApplicantImageNameChooser(grok.Adapter):
360    """A file id chooser for :class:`Applicant` objects.
361
362    `context` is an :class:`Applicant` instance.
363
364    The :class:`ApplicantImageNameChooser` can build/check file ids
365    for :class:`Applicant` objects suitable for use with
366    :class:`ExtFileStore` instances. The delivered file_id contains
367    the file id marker for :class:`Applicant` object and the
368    registration number or access code of the context applicant. Also
369    the name of the connected applicant container will be part of the
370    generated file id.
371
372    This chooser is registered as an adapter providing
[7811]373    :class:`waeup.kofa.interfaces.IFileStoreNameChooser`.
[7063]374
375    File store name choosers like this one are only convenience
376    components to ease the task of creating file ids for applicant
377    objects. You are nevertheless encouraged to use them instead of
378    manually setting up filenames for applicants.
379
[7811]380    .. seealso:: :mod:`waeup.kofa.imagestorage`
[7063]381
382    """
383    grok.context(IApplicant)
384    grok.implements(IFileStoreNameChooser)
385
[7067]386    def checkName(self, name=None, attr=None):
[7063]387        """Check whether the given name is a valid file id for the context.
388
389        Returns ``True`` only if `name` equals the result of
390        :meth:`chooseName`.
[7067]391
392        The `attr` parameter is not taken into account for
393        :class:`Applicant` context as the single passport image is the
394        only file we store for applicants.
[7063]395        """
396        return name == self.chooseName()
397
[7105]398    def chooseName(self, name=None, attr=None):
[7063]399        """Get a valid file id for applicant context.
400
401        *Example:*
402
[7240]403        For an applicant with applicant_id. ``'app2001_1234'``
[7063]404        and stored in an applicants container called
405        ``'mycontainer'``, this chooser would create:
406
[7240]407          ``'__img-applicant__mycontainer/app2001_1234.jpg'``
[7063]408
409        meaning that the passport image of this applicant would be
410        stored in the site-wide file storage in path:
411
[7240]412          ``mycontainer/app2001_1234.jpg``
[7063]413
414        If the context applicant has no parent, ``'_default'`` is used
415        as parent name.
[7067]416
[10089]417        In the beginning the `attr` parameter was not taken into account for
418        :class:`Applicant` context as the single passport image was the
[15941]419        only file we store for applicants. Meanwhile many universities require
[10089]420        uploads of other documents too. Now we store passport image
421        files without attribute but all other documents with.
[7067]422
[7063]423        """
424        parent_name = getattr(
425            getattr(self.context, '__parent__', None),
426            '__name__', '_default')
[10089]427        if attr is None or attr == 'passport.jpg':
428            marked_filename = '__%s__%s/%s.jpg' % (
429                APPLICANT_IMAGE_STORE_NAME,
430                parent_name, self.context.applicant_id)
431        else:
432            basename, ext = os.path.splitext(attr)
433            marked_filename = '__%s__%s/%s_%s%s' % (
434                APPLICANT_IMAGE_STORE_NAME,
[10092]435                parent_name, basename, self.context.applicant_id, ext)
[7063]436        return marked_filename
437
438
439class ApplicantImageStoreHandler(DefaultFileStoreHandler, grok.GlobalUtility):
440    """Applicant specific image handling.
441
442    This handler knows in which path in a filestore to store applicant
443    images and how to turn this kind of data into some (browsable)
444    file object.
445
446    It is called from the global file storage, when it wants to
447    get/store a file with a file id starting with
448    ``__img-applicant__`` (the marker string for applicant images).
449
450    Like each other file store handler it does not handle the files
451    really (this is done by the global file store) but only computes
452    paths and things like this.
453    """
454    grok.implements(IFileStoreHandler)
455    grok.name(APPLICANT_IMAGE_STORE_NAME)
456
457    def pathFromFileID(self, store, root, file_id):
458        """All applicants images are filed in directory ``applicants``.
459        """
460        marker, filename, basename, ext = store.extractMarker(file_id)
[7121]461        sub_root = os.path.join(root, 'applicants')
462        return super(ApplicantImageStoreHandler, self).pathFromFileID(
463            store, sub_root, basename)
[7063]464
465    def createFile(self, store, root, filename, file_id, file):
466        """Create a browsable file-like object.
467        """
[7121]468        # call super method to ensure that any old files with
469        # different filename extension are deleted.
470        file, path, file_obj =  super(
471            ApplicantImageStoreHandler, self).createFile(
472            store, root,  filename, file_id, file)
[7819]473        return file, path, KofaImageFile(
[7121]474            file_obj.filename, file_obj.data)
[7240]475
476@grok.subscribe(IApplicant, grok.IObjectAddedEvent)
477def handle_applicant_added(applicant, event):
478    """If an applicant is added local and site roles are assigned.
479    """
480    role_manager = IPrincipalRoleManager(applicant)
481    role_manager.assignRoleToPrincipal(
482        'waeup.local.ApplicationOwner', applicant.applicant_id)
483    # Assign current principal the global Applicant role
484    role_manager = IPrincipalRoleManager(grok.getSite())
485    role_manager.assignRoleToPrincipal(
486        'waeup.Applicant', applicant.applicant_id)
[8375]487    if applicant.state is None:
488        IWorkflowInfo(applicant).fireTransition('init')
[7240]489
490    # Assign global applicant role for new applicant (alternative way)
491    #account = IUserAccount(applicant)
492    #account.roles = ['waeup.Applicant']
493
494    return
495
496@grok.subscribe(IApplicant, grok.IObjectRemovedEvent)
497def handle_applicant_removed(applicant, event):
[13224]498    """If an applicant is removed a message is logged, passport images are
499    deleted and the global role is unset.
[7240]500    """
[13073]501    comment = 'Application record removed'
[7240]502    target = applicant.applicant_id
503    try:
[7652]504        grok.getSite()['applicants'].logger.info('%s - %s' % (
505            target, comment))
[7240]506    except KeyError:
507        # If we delete an entire university instance there won't be
508        # an applicants subcontainer
509        return
[13224]510    # Remove any passport image.
[8285]511    file_store = getUtility(IExtFileStore)
512    file_store.deleteFileByContext(applicant)
[15941]513    # Remove all other files too
[15943]514    filenames = getUtility(IApplicantsUtils).ADDITIONAL_FILES
515    for filename in filenames:
516        file_store.deleteFileByContext(applicant, attr=filename[1])
[13224]517    # Remove global role
518    role_manager = IPrincipalRoleManager(grok.getSite())
519    role_manager.unsetRoleForPrincipal(
520        'waeup.Applicant', applicant.applicant_id)
[7240]521    return
Note: See TracBrowser for help on using the repository browser.