source: main/waeup.sirp/trunk/src/waeup/sirp/applicants/interfaces.py @ 6706

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

Reorganize vocabularies. The vocabs of the students package are the basis. In the application package only additional sources and vocabs should be defined.

File size: 17.0 KB
RevLine 
[5638]1##
2## interfaces.py
3## Login : <uli@pu.smp.net>
4## Started on  Sun Jan 16 15:30:01 2011 Uli Fouquet
5## $Id$
[6076]6##
[6087]7## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
[5638]8## This program is free software; you can redistribute it and/or modify
9## it under the terms of the GNU General Public License as published by
10## the Free Software Foundation; either version 2 of the License, or
11## (at your option) any later version.
[6076]12##
[5638]13## This program is distributed in the hope that it will be useful,
14## but WITHOUT ANY WARRANTY; without even the implied warranty of
15## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16## GNU General Public License for more details.
[6076]17##
[5638]18## You should have received a copy of the GNU General Public License
19## along with this program; if not, write to the Free Software
20## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21##
[6500]22"""Interfaces of the university application package.
[5638]23"""
[5753]24import os
[6343]25import re
[5753]26import waeup.sirp.browser
[6256]27
[5866]28from grokcore.content.interfaces import IContainer
[6256]29
[5638]30from zope import schema
[6290]31from zope.interface import Interface, Attribute, provider
[6391]32from zope.component import getUtilitiesFor
[5758]33from zope.pluggableauth.interfaces import IPrincipalInfo
34from zope.security.interfaces import IGroupClosureAwarePrincipal as IPrincipal
[6256]35from zc.sourcefactory.basic import BasicSourceFactory
[6594]36from waeup.sirp.image import createWAeUPImageFile
[5753]37from waeup.sirp.image.schema import ImageFile
[6256]38from waeup.sirp.interfaces import IWAeUPObject
[6189]39from waeup.sirp.university.vocabularies import application_categories
[6648]40from waeup.sirp.students.vocabularies import (
41  year_range, lgas_vocab, CertificateSource, GenderSource,
42  )
[6256]43from waeup.sirp.applicants.vocabularies import (
[6648]44  application_types_vocab, application_pins_vocab,
45  AppCatCertificateSource,
[6256]46  )
[5638]47
[5753]48IMAGE_PATH = os.path.join(
49    os.path.dirname(waeup.sirp.browser.__file__),
50    'static'
51    )
[6313]52DEFAULT_PASSPORT_IMAGE_MALE = open(
53    os.path.join(IMAGE_PATH, 'placeholder_m.jpg')).read()
54DEFAULT_PASSPORT_IMAGE_FEMALE = open(
55    os.path.join(IMAGE_PATH, 'placeholder_f.jpg')).read()
[5753]56
[6343]57# Define a valiation method for email addresses
58class NotAnEmailAddress(schema.ValidationError):
59    __doc__ = u"Invalid email address"
60
[6594]61check_email = re.compile(
62    r"[a-zA-Z0-9._%-]+@([a-zA-Z0-9-]+.)*[a-zA-Z]{2,4}").match
[6343]63def validate_email(value):
64    if not check_email(value):
65        raise NotAnEmailAddress(value)
66    return True
67
[6290]68@provider(schema.interfaces.IContextAwareDefaultFactory)
69def default_passport_image(context):
70    """A default value factory for ImageFile fields.
71
72    Returns some default image as WAeUPImageFile. We cannot set the
73    default directly in ImageFile fields, as, if we want to set
74    max_size or min_size as well, some utility lookups are needed
75    which are not possible during startup.
76
77    Developers which use IContextAwareDefaultFactories like this one
78    always should make sure that the delivered default meets all
79    constraints of the field that makes use of this default value
80    provider.
81    """
[6538]82    imagefile = createWAeUPImageFile(
83        'placeholder_m.jpg',
84        open(os.path.join(IMAGE_PATH, 'placeholder_m.jpg'), 'r')
85        )
86    return imagefile
[6290]87
[6069]88class ApplicantContainerProviderSource(BasicSourceFactory):
[6075]89    """A source offering all available applicants container types.
90
91    The values returned by this source are names of utilities that can
92    create :class:`ApplicantContainer` instances. So, if you get a
93    name like ``'myactype'`` from this source, then you can do:
94
95      >>> from zope.component import getUtility
96      >>> p = getUtility(IApplicantsContainerProvider, name=myactype)
97      >>> my_applicants_container = p.factory()
98
99    Or you can access class-attributes like
100
101      >>> my_applicants_container.container_title
102      'Pretty'
[6076]103
[6069]104    """
105    def getValues(self):
[6075]106        """Returns a list of ``(<name>, <provider>)`` tuples.
[5753]107
[6075]108        Here ``<name>`` is the name under which an
109        :class:``ApplicantContainerProvider`` was registered as a
110        utility and ``<provider>`` is the utility itself.
111        """
112        return getUtilitiesFor(IApplicantsContainerProvider)
113
[6069]114    def getToken(self, value):
[6075]115        """Return the name of the ``(<name>, <provider>)`` tuple.
116        """
117        return value[0]
118
[6069]119    def getTitle(self, value):
[6075]120        """Get a 'title - description' string for a container type.
121        """
122        factory = value[1].factory
123        return "%s - %s" % (
124            factory.container_title, factory.container_description)
[6069]125
[5753]126class IResultEntry(IWAeUPObject):
127    subject = schema.TextLine(
128        title = u'Subject',
129        description = u'The subject',
130        required=False,
131        )
132    score = schema.TextLine(
133        title = u'Score',
134        description = u'The score',
135        required=False,
136        )
137
[5866]138class IApplicantsRoot(IWAeUPObject, IContainer):
[5676]139    """A container for university applicants containers.
[5645]140    """
[5866]141    pass
[5638]142
[5676]143class IApplicantsContainer(IWAeUPObject):
144    """An applicants container contains university applicants.
[5645]145
[5638]146    """
[6069]147
[6075]148    container_title = Attribute(
149        u'classattribute: title for type of container')
150    container_description = Attribute(
151        u'classattribute: description for type of container')
152
[6076]153
[6069]154    code = schema.TextLine(
155        title = u'Code',
[6087]156        default = u'-',
[6069]157        required = True,
158        readonly = True,
[6076]159        )
[6096]160
[6087]161    title = schema.TextLine(
162        title = u'Title',
163        required = True,
164        default = u'-',
165        readonly = True,
[6096]166        )
167
[6087]168    prefix = schema.Choice(
169        title = u'Application target',
170        required = True,
171        default = None,
172        source = application_types_vocab,
173        readonly = True,
174        )
[6076]175
[6087]176    year = schema.Choice(
177        title = u'Year of entrance',
178        required = True,
179        default = None,
[6158]180        values = year_range(),
[6087]181        readonly = True,
[6096]182        )
[6087]183
[6069]184    provider = schema.Choice(
[6070]185        title = u'Applicants container type',
[6069]186        required = True,
187        default = None,
188        source = ApplicantContainerProviderSource(),
[6087]189        readonly = True,
[6076]190        )
[6158]191
[6110]192    ac_prefix = schema.Choice(
193        title = u'Access code prefix',
194        required = True,
195        default = None,
[6111]196        source = application_pins_vocab,
[6110]197        )
[6076]198
[6189]199    application_category = schema.Choice(
[6477]200        title = u'Category for the grouping of certificates',
[6189]201        required = True,
202        default = None,
203        source = application_categories,
204        )
205
[5645]206    description = schema.Text(
207        title = u'Human readable description in reST format',
[5638]208        required = False,
[6518]209        default = u'''This text can been seen by anonymous users.
210Here we put information about the study courses provided, the application procedure and deadlines.'''
[5638]211        )
212
213    startdate = schema.Date(
[6509]214        title = u'Application start date',
[5638]215        required = False,
216        default = None,
217        )
218
219    enddate = schema.Date(
[6509]220        title = u'Application closing date',
[5638]221        required = False,
222        default = None,
223        )
224
[5645]225    strict_deadline = schema.Bool(
226        title = u'Forbid additions after deadline (enddate)',
227        required = True,
228        default = True,
229        )
[5638]230
231    def archive(id=None):
[5676]232        """Create on-dist archive of applicants stored in this term.
[5638]233
[5676]234        If id is `None`, all applicants are archived.
[5638]235
236        If id contains a single id string, only the respective
[5676]237        applicants are archived.
[5638]238
239        If id contains a list of id strings all of the respective
[5676]240        applicants types are saved to disk.
[5638]241        """
242
243    def clear(id=None, archive=True):
[5676]244        """Remove applicants of type given by 'id'.
[5638]245
[5676]246        Optionally archive the applicants.
[6076]247
[5676]248        If id is `None`, all applicants are archived.
[5638]249
250        If id contains a single id string, only the respective
[5676]251        applicants are archived.
[5638]252
253        If id contains a list of id strings all of the respective
[5676]254        applicant types are saved to disk.
[5638]255
256        If `archive` is ``False`` none of the archive-handling is done
[5676]257        and respective applicants are simply removed from the
[5638]258        database.
259        """
[6073]260
[6069]261class IApplicantsContainerAdd(IApplicantsContainer):
262    """An applicants container contains university applicants.
263    """
[6087]264    prefix = schema.Choice(
265        title = u'Application target',
[6069]266        required = True,
[6087]267        default = None,
268        source = application_types_vocab,
[6069]269        readonly = False,
[6076]270        )
[6073]271
[6087]272    year = schema.Choice(
273        title = u'Year of entrance',
274        required = True,
275        default = None,
[6158]276        values = year_range(),
[6087]277        readonly = False,
[6096]278        )
[6073]279
[6087]280    provider = schema.Choice(
281        title = u'Applicants container type',
282        required = True,
283        default = None,
284        source = ApplicantContainerProviderSource(),
285        readonly = False,
286        )
287
[6096]288IApplicantsContainerAdd[
289    'prefix'].order =  IApplicantsContainer['prefix'].order
290IApplicantsContainerAdd[
291    'year'].order =  IApplicantsContainer['year'].order
292IApplicantsContainerAdd[
293    'provider'].order =  IApplicantsContainer['provider'].order
[6087]294
[5753]295class IApplicantBaseData(IWAeUPObject):
296    """The data for an applicant.
297
298    This is a base interface with no field (except ``reg_no``)
299    required. For use with importers, forms, etc., please use one of
300    the derived interfaces below, which set more fields to required
301    state, depending on use-case.
302    """
[6339]303    history = Attribute('Object history, a list of messages.')
[6332]304    state = Attribute('Returns the application state of an applicant')
[6476]305    application_date = Attribute('Date of submission, used for export only')
[6304]306
[6476]307    #def getApplicantsRootLogger():
308    #    """Returns the logger from the applicants root object
309    #    """
310
311    def loggerInfo(ob_class, comment):
312        """Adds an INFO message to the log file
[6348]313        """
314
[5753]315    reg_no = schema.TextLine(
316        title = u'JAMB Registration Number',
[6195]317        readonly = True,
[5753]318        )
319    access_code = schema.TextLine(
320        title = u'Access Code',
321        required = False,
[6195]322        readonly = True,
[5753]323        )
[6248]324    course1 = schema.Choice(
[5753]325        title = u'1st Choice Course of Study',
[6248]326        source = AppCatCertificateSource(),
[6352]327        required = True,
[5753]328        )
[6248]329    course2 = schema.Choice(
[5753]330        title = u'2nd Choice Course of Study',
[6248]331        source = AppCatCertificateSource(),
[5753]332        required = False,
333        )
334    firstname = schema.TextLine(
335        title = u'First Name',
[6352]336        required = True,
[5753]337        )
338    middlenames = schema.TextLine(
339        title = u'Middle Names',
340        required = False,
341        )
342    lastname = schema.TextLine(
[6205]343        title = u'Last Name (Surname)',
[6352]344        required = True,
[5753]345        )
346    date_of_birth = schema.Date(
347        title = u'Date of Birth',
[6352]348        required = True,
[5753]349        )
[6249]350    lga = schema.Choice(
351        source = lgas_vocab,
[6205]352        title = u'State/LGA',
[6254]353        default = 'foreigner',
354        required = True,
[5753]355        )
356    sex = schema.Choice(
357        title = u'Sex',
358        source = GenderSource(),
359        default = u'm',
[6352]360        required = True,
[5753]361        )
[6341]362    email = schema.ASCIILine(
[5753]363        title = u'Email',
364        required = False,
[6343]365        constraint=validate_email,
[5753]366        )
[6341]367    phone = schema.Int(
[5753]368        title = u'Phone',
[6341]369        description = u'Enter phone number with country code and without spaces.',
[5753]370        required = False,
371        )
372    passport = ImageFile(
373        title = u'Passport Photograph',
[6290]374        #default = DEFAULT_PASSPORT_IMAGE_MALE,
375        defaultFactory = default_passport_image,
[6341]376        description = u'Maximun file size is 20 kB.',
[5753]377        required = True,
[6285]378        max_size = 20480,
[5753]379        )
[6322]380
[5753]381    #
[6195]382    # Process Data
[5753]383    #
[6255]384    screening_score = schema.Int(
[5753]385        title = u'Screening Score',
386        required = False,
387        )
388    screening_venue = schema.TextLine(
389        title = u'Screening Venue',
390        required = False,
391        )
[6248]392    course_admitted = schema.Choice(
[5753]393        title = u'Admitted Course of Study',
[6248]394        source = CertificateSource(),
[6254]395        default = None,
[5753]396        required = False,
397        )
398    notice = schema.Text(
399        title = u'Notice',
400        required = False,
401        )
402    student_id = schema.TextLine(
403        title = u'Student ID',
404        required = False,
[6195]405        readonly = True,
[5753]406        )
[6302]407    locked = schema.Bool(
408        title = u'Form locked',
409        default = False,
410        )
[5753]411
412class IApplicant(IApplicantBaseData):
413    """An applicant.
414
415    This is basically the applicant base data. Here we repeat the
[6195]416    fields from base data if we have to set the `required` attribute
417    to True (which is the default).
[5753]418    """
419
[6195]420class IApplicantEdit(IApplicantBaseData):
421    """An applicant.
[5753]422
[6339]423    Here we can repeat the fields from base data and set the
424    `required` and `readonly` attributes to True to further restrict
425    the data access. We cannot omit fields. This has to be done in the
426    respective form page.
[6195]427    """
[6255]428    screening_score = schema.Int(
[5753]429        title = u'Screening Score',
[6195]430        required = False,
[5941]431        readonly = True,
432        )
[6195]433    screening_venue = schema.TextLine(
434        title = u'Screening Venue',
[5753]435        required = False,
436        readonly = True,
437        )
[6301]438    course_admitted = schema.Choice(
[6195]439        title = u'Admitted Course of Study',
[6301]440        source = CertificateSource(),
441        default = None,
[5753]442        required = False,
[6195]443        readonly = True,
[5753]444        )
[6337]445    # entry_session is inherited from the container
446    #entry_session = schema.Choice(
447    #    source = entry_session_vocab,
448    #    title = u'Entry Session',
449    #    required = False,
450    #    readonly = True
451    #    )
[6195]452    notice = schema.Text(
453        title = u'Notice',
[5753]454        required = False,
455        readonly = True,
456        )
[5758]457
458class IApplicantPrincipalInfo(IPrincipalInfo):
459    """Infos about principals that are applicants.
460    """
461    access_code = Attribute("The Access Code the user purchased")
462
463class IApplicantPrincipal(IPrincipal):
464    """A principal that is an applicant.
465
466    This interface extends zope.security.interfaces.IPrincipal and
467    requires also an `id` and other attributes defined there.
468    """
469    access_code = schema.TextLine(
470        title = u'Access Code',
471        description = u'The access code purchased by the user.',
472        required = True,
473        readonly = True)
474
475class IApplicantsFormChallenger(Interface):
476    """A challenger that uses a browser form to collect applicant
477       credentials.
478    """
479    loginpagename = schema.TextLine(
480        title = u'Loginpagename',
481        description = u"""Name of the login form used by challenger.
482
483        The form must provide an ``access_code`` input field.
484        """)
485
486    accesscode_field = schema.TextLine(
487        title = u'Access code field',
488        description = u'''Field of the login page which is looked up for
489                          access_code''',
490        default = u'access_code',
491        )
492
493
494class IApplicantSessionCredentials(Interface):
495    """Interface for storing and accessing applicant credentials in a
496       session.
497    """
498
499    def __init__(access_code):
500        """Create applicant session credentials."""
501
502    def getAccessCode():
503        """Return the access code."""
504
505
[5846]506class IApplicantsContainerProvider(Interface):
[5820]507    """A provider for applicants containers.
508
509    Applicants container providers are meant to be looked up as
510    utilities. This way we can find all applicant container types
511    defined somewhere.
512
513    Each applicants container provider registered as utility provides
514    one container type and one should be able to call the `factory`
515    attribute to create an instance of the requested container type.
516
517    .. THE FOLLOWING SHOULD GO INTO SPHINX DOCS (and be tested)
[6076]518
[6500]519    Samples:
[6076]520
[5820]521    Given, you had an IApplicantsContainer implementation somewhere
522    and you would like to make it findable on request, then you would
523    normally create an appropriate provider utility like this::
524
525      import grok
[5846]526      from waeup.sirp.applicants.interfaces import IApplicantsContainerProvider
[5820]527
528      class MyContainerProvider(grok.GlobalUtility):
[5846]529          grok.implements(IApplicantsContainerProvider)
[5820]530          grok.name('MyContainerProvider') # Must be unique
531          factory = MyContainer # A class implementing IApplicantsContainer
532                                # or derivations thereof.
533
534    This utility would be registered on startup and could then be used
535    like this:
536
537      >>> from zope.component import getAllUtilitiesRegisteredFor
538      >>> from waeup.sirp.applicants.interfaces import (
[5846]539      ...     IApplicantsContainerProvider)
[5820]540      >>> all_providers = getAllUtilitiesRegisteredFor(
[5846]541      ...     IApplicantsContainerProvider)
[5820]542      >>> all_providers
543      [<MyContainerProvider object at 0x...>]
544
545    You could look up this specific provider by name:
546
547      >>> from zope.component import getUtility
[5846]548      >>> p = getUtility(IApplicantsContainerProvider, name='MyProvider')
[5820]549      >>> p
550      <MyContainerProvider object at 0x...>
[6076]551
[5820]552    An applicants container would then be created like this:
553
554      >>> provider = all_providers[0]
555      >>> container = provider.factory()
556      >>> container
557      <MyContainer object at 0x...>
558
559    """
560    factory = Attribute("A class that can create instances of the "
561                        "requested container type")
Note: See TracBrowser for help on using the repository browser.