source: main/waeup.kofa/trunk/src/waeup/kofa/interfaces.py @ 16495

Last change on this file since 16495 was 16372, checked in by Henrik Bettermann, 4 years ago

Extend academic_sessions range.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 46.0 KB
Line 
1## $Id: interfaces.py 16372 2021-01-13 18:55:42Z 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##
18import os
19import re
20import codecs
21import zc.async.interfaces
22import zope.i18nmessageid
23from datetime import datetime
24from hurry.file.interfaces import IFileRetrieval
25from hurry.workflow.interfaces import IWorkflowInfo
26from zc.sourcefactory.basic import BasicSourceFactory
27from zope import schema
28from zope.pluggableauth.interfaces import IPrincipalInfo
29from zope.security.interfaces import IGroupClosureAwarePrincipal as IPrincipal
30from zope.component import getUtility
31from zope.component.interfaces import IObjectEvent
32from zope.configuration.fields import Path
33from zope.container.interfaces import INameChooser, IContainer
34from zope.interface import Interface, Attribute
35from zope.schema.interfaces import IObject
36from zope.schema.vocabulary import SimpleVocabulary, SimpleTerm
37from waeup.kofa.schema import PhoneNumber
38from waeup.kofa.sourcefactory import SmartBasicContextualSourceFactory
39
40_ = MessageFactory = zope.i18nmessageid.MessageFactory('waeup.kofa')
41
42DELETION_MARKER = 'XXX'
43IGNORE_MARKER = '<IGNORE>'
44WAEUP_KEY = 'waeup.kofa'
45VIRT_JOBS_CONTAINER_NAME = 'jobs'
46DOCLINK = 'http://kofa-doc.waeup.org/userdocs'
47
48CREATED = 'created'
49ADMITTED = 'admitted'
50CLEARANCE = 'clearance started'
51REQUESTED = 'clearance requested'
52CLEARED = 'cleared'
53PAID = 'school fee paid'
54RETURNING = 'returning'
55REGISTERED = 'courses registered'
56VALIDATED = 'courses validated'
57GRADUATED = 'graduated'
58TRANSREQ = 'transcript requested'
59TRANSVAL = 'transcript validated'
60TRANSREL = 'transcript released'
61
62
63#: A dict giving job status as tuple (<STRING>, <TRANSLATED_STRING>),
64#: the latter for UI purposes.
65JOB_STATUS_MAP = {
66    zc.async.interfaces.NEW: ('new', _('new')),
67    zc.async.interfaces.COMPLETED: ('completed', _('completed')),
68    zc.async.interfaces.PENDING: ('pending', _('pending')),
69    zc.async.interfaces.ACTIVE: ('active', _('active')),
70    zc.async.interfaces.ASSIGNED: ('assigned', _('assigned')),
71    zc.async.interfaces.CALLBACKS: ('callbacks', _('callbacks')),
72    }
73
74#default_rest_frontpage = u'' + codecs.open(os.path.join(
75#        os.path.dirname(__file__), 'frontpage.rst'),
76#        encoding='utf-8', mode='rb').read()
77
78default_html_frontpage = u'' + codecs.open(os.path.join(
79        os.path.dirname(__file__), 'frontpage.html'),
80        encoding='utf-8', mode='rb').read()
81
82def SimpleKofaVocabulary(*terms):
83    """A well-buildt vocabulary provides terms with a value, token and
84       title for each term
85    """
86    return SimpleVocabulary([
87            SimpleTerm(value, value, title) for title, value in terms])
88
89def academic_sessions():
90    curr_year = datetime.now().year
91    year_range = range(1970, curr_year + 2)
92    return [('%s/%s' % (year,year+1), year) for year in year_range]
93
94academic_sessions_vocab = SimpleKofaVocabulary(*academic_sessions())
95
96registration_states_vocab = SimpleKofaVocabulary(
97    (_('created'), CREATED),
98    (_('admitted'), ADMITTED),
99    (_('clearance started'), CLEARANCE),
100    (_('clearance requested'), REQUESTED),
101    (_('cleared'), CLEARED),
102    (_('school fee paid'), PAID),
103    (_('courses registered'), REGISTERED),
104    (_('courses validated'), VALIDATED),
105    (_('returning'), RETURNING),
106    (_('graduated'), GRADUATED),
107    (_('transcript requested'), TRANSREQ),
108    (_('transcript validated'), TRANSVAL),
109    (_('transcript released'), TRANSREL),
110    )
111
112class ContextualDictSourceFactoryBase(SmartBasicContextualSourceFactory):
113    """A base for contextual sources based on KofaUtils dicts.
114
115    To create a real source, you have to set the `DICT_NAME` attribute
116    which should be the name of a dictionary in KofaUtils.
117    """
118    def getValues(self, context):
119        utils = getUtility(IKofaUtils)
120        sorted_items = sorted(getattr(utils, self.DICT_NAME).items(),
121                              key=lambda item: item[1])
122        return [item[0] for item in sorted_items]
123
124    def getToken(self, context, value):
125        return str(value)
126
127    def getTitle(self, context, value):
128        utils = getUtility(IKofaUtils)
129        return getattr(utils, self.DICT_NAME)[value]
130
131class SubjectSource(BasicSourceFactory):
132    """A source for school subjects used in exam documentation.
133    """
134    def getValues(self):
135        subjects_dict = getUtility(IKofaUtils).EXAM_SUBJECTS_DICT
136        return sorted(subjects_dict.keys())
137
138    def getTitle(self, value):
139        subjects_dict = getUtility(IKofaUtils).EXAM_SUBJECTS_DICT
140        return "%s:" % subjects_dict[value]
141
142class GradeSource(BasicSourceFactory):
143    """A source for exam grades.
144    """
145    def getValues(self):
146        for entry in getUtility(IKofaUtils).EXAM_GRADES:
147            yield entry[0]
148
149    def getTitle(self, value):
150        return dict(getUtility(IKofaUtils).EXAM_GRADES)[value]
151
152class DisablePaymentGroupSource(ContextualDictSourceFactoryBase):
153    """A source for filtering groups of students
154    """
155    #: name of dict to deliver from kofa utils.
156    DICT_NAME = 'DISABLE_PAYMENT_GROUP_DICT'
157
158# Define a validation method for email addresses
159class NotAnEmailAddress(schema.ValidationError):
160    __doc__ = u"Invalid email address"
161
162#: Regular expression to check email-address formats. As these can
163#: become rather complex (nearly everything is allowed by RFCs), we only
164#: forbid whitespaces, commas and dots following onto each other.
165check_email = re.compile(
166    r"^[^@\s,]+@[^@\.\s,]+(\.[^@\.\s,]+)*$").match
167
168def validate_email(value):
169    if not check_email(value):
170        raise NotAnEmailAddress(value)
171    return True
172
173# Define a validation method for ids
174class NotIdValue(schema.ValidationError):
175    __doc__ = u"Invalid id"
176
177#: Regular expressions to check id formats.
178check_id = re.compile(r"^[a-zA-Z0-9_-]{2,11}$").match
179
180def validate_id(value):
181    if not check_id(value):
182        raise NotIdValue(value)
183    return True
184
185# Define a validation method for HTML fields
186class NotHTMLValue(schema.ValidationError):
187    __doc__ = u"Style or script elements not allowed"
188
189def validate_html(value):
190    if '<style' in value or '<script' in value:
191        raise NotHTMLValue(value)
192    return True
193
194# Define a validation method for international phone numbers
195class InvalidPhoneNumber(schema.ValidationError):
196    __doc__ = u"Invalid phone number"
197
198# represent format +NNN-NNNN-NNNN
199RE_INT_PHONE = re.compile(r"^\+?\d+\-\d+\-[\d\-]+$")
200
201def validate_phone(value):
202    if not RE_INT_PHONE.match(value):
203        raise InvalidPhoneNumber(value)
204    return True
205
206class FatalCSVError(Exception):
207    """Some row could not be processed.
208    """
209    pass
210
211class DuplicationError(Exception):
212    """An exception that can be raised when duplicates are found.
213
214    When raising :exc:`DuplicationError` you can, beside the usual
215    message, specify a list of objects which are duplicates. These
216    values can be used by catching code to print something helpful or
217    similar.
218    """
219    def __init__(self, msg, entries=[]):
220        self.msg = msg
221        self.entries = entries
222
223    def __str__(self):
224        return '%r' % self.msg
225
226class RoleSource(BasicSourceFactory):
227    """A source for site roles.
228    """
229    def getValues(self):
230        # late import: in interfaces we should not import local modules
231        from waeup.kofa.permissions import get_waeup_role_names
232        return get_waeup_role_names()
233
234    def getTitle(self, value):
235        # late import: in interfaces we should not import local modules
236        from waeup.kofa.permissions import get_all_roles
237        roles = dict(get_all_roles())
238        if value in roles.keys():
239            title = roles[value].title
240            if '.' in title:
241                title = title.split('.', 2)[1]
242        return title
243
244class CaptchaSource(BasicSourceFactory):
245    """A source for captchas.
246    """
247    def getValues(self):
248        captchas = ['No captcha', 'Testing captcha', 'ReCaptcha']
249        try:
250            # we have to 'try' because IConfiguration can only handle
251            # interfaces from w.k.interface.
252            from waeup.kofa.browser.interfaces import ICaptchaManager
253        except:
254            return captchas
255        return sorted(getUtility(ICaptchaManager).getAvailCaptchas().keys())
256
257    def getTitle(self, value):
258        return value
259
260class IResultEntry(Interface):
261    """A school grade entry.
262    """
263
264    subject = schema.Choice(
265        title = _(u'Subject'),
266        source = SubjectSource(),
267        )
268
269    grade = schema.Choice(
270        title = _(u'Grade'),
271        source = GradeSource(),
272        )
273
274class IResultEntryField(IObject):
275    """A zope.schema-like field for usage in interfaces.
276
277    Marker interface to distuingish result entries from ordinary
278    object fields. Needed for registration of widgets.
279    """
280
281class IRefereeEntry(Interface):
282    """A referee entry.
283    """
284    email_sent = Attribute('True if email has been sent')
285
286    name = schema.TextLine(
287        title = _(u'Name'),
288        required = True,
289        description = _(u'Name'),
290        )
291
292    email = schema.ASCIILine(
293        title = _(u'Email Address'),
294        default = None,
295        required = True,
296        constraint=validate_email,
297        description = _(
298            u'Email Address'),
299        )
300
301class IRefereeEntryField(IObject):
302    """A zope.schema-like field for usage in interfaces.
303
304    Marker interface to distuingish referee entries from ordinary
305    object fields. Needed for registration of widgets.
306    """
307
308class IKofaUtils(Interface):
309    """A collection of methods which are subject to customization.
310    """
311
312    PORTAL_LANGUAGE = Attribute("Dict of global language setting")
313    PREFERRED_LANGUAGES_DICT = Attribute("Dict of preferred languages")
314    EXAM_SUBJECTS_DICT = Attribute("Dict of examination subjects")
315    EXAM_GRADES = Attribute("Dict of examination grades")
316    INST_TYPES_DICT = Attribute("Dict if institution types")
317    STUDY_MODES_DICT = Attribute("Dict of study modes")
318    APP_CATS_DICT = Attribute("Dict of application categories")
319    SEMESTER_DICT = Attribute("Dict of semesters or trimesters")
320    SYSTEM_MAX_LOAD = Attribute("Dict of maximum system loads.")
321    TEMP_PASSWORD_MINUTES = Attribute(
322        "Temporary passwords and parents password validity period")
323
324    def sendContactForm(
325          from_name,from_addr,rcpt_name,rcpt_addr,
326          from_username,usertype,portal,body,subject):
327        """Send an email with data provided by forms.
328        """
329
330    def fullname(firstname,lastname,middlename):
331        """Full name constructor.
332        """
333
334    def sendCredentials(user, password, url_info, msg):
335        """Send credentials as email.
336
337        Input is the applicant for which credentials are sent and the
338        password.
339
340        Returns True or False to indicate successful operation.
341        """
342
343    def genPassword(length, chars):
344        """Generate a random password.
345        """
346
347class IKofaObject(Interface):
348    """A Kofa object.
349
350    This is merely a marker interface.
351    """
352
353class IUniversity(IKofaObject):
354    """Representation of a university.
355    """
356
357
358class IKofaContainer(IKofaObject):
359    """A container for Kofa objects.
360    """
361
362class IKofaContained(IKofaObject):
363    """An item contained in an IKofaContainer.
364    """
365
366class ICSVExporter(Interface):
367    """A CSV file exporter for objects.
368    """
369    fields = Attribute("""List of fieldnames in resulting CSV""")
370
371    title = schema.TextLine(
372        title = u'Title',
373        description = u'Description to be displayed in selections.',
374        )
375    def mangle_value(value, name, obj):
376        """Mangle `value` extracted from `obj` or suobjects thereof.
377
378        This is called by export before actually writing to the result
379        file.
380        """
381
382    def get_filtered(site, **kw):
383        """Get datasets in `site` to be exported.
384
385        The set of data is specified by keywords, which might be
386        different for any implementaion of exporter.
387
388        Returns an iterable.
389        """
390
391    def get_selected(site, selected):
392        """Get datasets in `site` to be exported.
393
394        The set of data is specified by a list of identifiers.
395
396        Returns an iterable.
397        """
398
399    def export(iterable, filepath=None):
400        """Export iterables as rows in a CSV file.
401
402        If `filepath` is not given, a string with the data should be
403        returned.
404
405        What kind of iterables are acceptable depends on the specific
406        exporter implementation.
407        """
408
409    def export_all(site, filepath=None):
410        """Export all items in `site` as CSV file.
411
412        if `filepath` is not given, a string with the data should be
413        returned.
414        """
415
416    def export_filtered(site, filepath=None, **kw):
417        """Export those items in `site` specified by `args` and `kw`.
418
419        If `filepath` is not given, a string with the data should be
420        returned.
421
422        Which special keywords are supported is up to the respective
423        exporter.
424        """
425
426    def export_selected(site, filepath=None, **kw):
427        """Export items in `site` specified by a list of identifiers
428        called `selected`.
429
430        If `filepath` is not given, a string with the data should be
431        returned.
432        """
433
434class IKofaExporter(Interface):
435    """An exporter for objects.
436    """
437    def export(obj, filepath=None):
438        """Export by pickling.
439
440        Returns a file-like object containing a representation of `obj`.
441
442        This is done using `pickle`. If `filepath` is ``None``, a
443        `cStringIO` object is returned, that contains the saved data.
444        """
445
446class IKofaXMLExporter(Interface):
447    """An XML exporter for objects.
448    """
449    def export(obj, filepath=None):
450        """Export as XML.
451
452        Returns an XML representation of `obj`.
453
454        If `filepath` is ``None``, a StringIO` object is returned,
455        that contains the transformed data.
456        """
457
458class IKofaXMLImporter(Interface):
459    """An XML import for objects.
460    """
461    def doImport(filepath):
462        """Create Python object from XML.
463
464        Returns a Python object.
465        """
466
467class IBatchProcessor(Interface):
468    """A batch processor that handles mass-operations.
469    """
470    name = schema.TextLine(
471        title = _(u'Processor name')
472        )
473
474    def doImport(path, headerfields, mode='create', user='Unknown',
475                 logger=None, ignore_empty=True):
476        """Read data from ``path`` and update connected object.
477
478        `headerfields` is a list of headerfields as read from the file
479        to import.
480
481        `mode` gives the import mode to use (``'create'``,
482        ``'update'``, or ``'remove'``.
483
484        `user` is a string describing the user performing the
485        import. Normally fetched from current principal.
486
487        `logger` is the logger to use during import.
488
489        `ignore_emtpy` in update mode ignores empty fields if true.
490        """
491
492class IContactForm(IKofaObject):
493    """A contact form.
494    """
495
496    email_from = schema.ASCIILine(
497        title = _(u'Email Address:'),
498        default = None,
499        required = True,
500        constraint=validate_email,
501        )
502
503    email_to = schema.ASCIILine(
504        title = _(u'Email to:'),
505        default = None,
506        required = True,
507        constraint=validate_email,
508        )
509
510    bcc_to = schema.Text(
511        title = _(u'Bcc to:'),
512        required = True,
513        #readonly = True,
514        )
515
516    subject = schema.TextLine(
517        title = _(u'Subject:'),
518        required = True,)
519
520    fullname = schema.TextLine(
521        title = _(u'Full Name:'),
522        required = True,)
523
524    body = schema.Text(
525        title = _(u'Text:'),
526        required = True,)
527
528
529class IKofaPrincipalInfo(IPrincipalInfo):
530    """Infos about principals that are users of Kofa Kofa.
531    """
532    email = Attribute("The email address of a user")
533    phone = Attribute("The phone number of a user")
534    public_name = Attribute("The public name of a user")
535    user_type = Attribute("The type of a user")
536
537
538class IKofaPrincipal(IPrincipal):
539    """A principle for Kofa Kofa.
540
541    This interface extends zope.security.interfaces.IPrincipal and
542    requires also an `id` and other attributes defined there.
543    """
544
545    email = schema.TextLine(
546        title = _(u'Email Address'),
547        description = u'',
548        required=False,)
549
550    phone = PhoneNumber(
551        title = _(u'Phone'),
552        description = u'',
553        required=False,)
554
555    public_name = schema.TextLine(
556        title = _(u'Public Name'),
557        required = False,)
558
559    user_type = Attribute('The user type of the principal')
560
561class IFailedLoginInfo(IKofaObject):
562    """Info about failed logins.
563
564    Timestamps are supposed to be stored as floats using time.time()
565    or similar.
566    """
567    num = schema.Int(
568        title = _(u'Number of failed logins'),
569        description = _(u'Number of failed logins'),
570        required = True,
571        default = 0,
572        )
573
574    last = schema.Float(
575        title = _(u'Timestamp'),
576        description = _(u'Timestamp of last failed login or `None`'),
577        required = False,
578        default = None,
579        )
580
581    def as_tuple():
582        """Get login info as tuple ``<NUM>, <TIMESTAMP>``.
583        """
584
585    def set_values(num=0, last=None):
586        """Set number of failed logins and timestamp of last one.
587        """
588
589    def increase():
590        """Increase the current number of failed logins and set timestamp.
591        """
592
593    def reset():
594        """Set failed login counters back to zero.
595        """
596
597
598class IUserAccount(IKofaObject):
599    """A user account.
600    """
601
602    failed_logins = Attribute('FailedLoginInfo for this account')
603
604    name = schema.TextLine(
605        title = _(u'User Id'),
606        description = _(u'Login name of user'),
607        required = True,)
608
609    title = schema.TextLine(
610        title = _(u'Full Name'),
611        required = True,)
612
613    public_name = schema.TextLine(
614        title = _(u'Public Name'),
615        description = _(u"Substitute for officer's real name "
616                       "in student object histories."),
617        required = False,)
618
619    description = schema.Text(
620        title = _(u'Description/Notice'),
621        required = False,)
622
623    email = schema.ASCIILine(
624        title = _(u'Email Address'),
625        default = None,
626        required = True,
627        constraint=validate_email,
628        )
629
630    phone = PhoneNumber(
631        title = _(u'Phone'),
632        default = None,
633        required = False,
634        )
635
636    roles = schema.List(
637        title = _(u'Portal Roles'),
638        value_type = schema.Choice(source=RoleSource()),
639        required = False,
640        )
641
642    suspended = schema.Bool(
643        title = _(u'Account suspended'),
644        description = _(u'If set, the account is immediately blocked.'),
645        default = False,
646        required = False,
647        )
648
649
650class IPasswordValidator(Interface):
651    """A password validator utility.
652    """
653
654    def validate_password(password, password_repeat):
655        """Validates a password by comparing it with
656        control password and checking some other requirements.
657        """
658
659    def validate_secure_password(self, pw, pw_repeat):
660        """ Validates a password by comparing it with
661        control password and checks password strength by
662        matching with the regular expression:
663
664        ^(?=.*[A-Z])(?=.*[a-z])(?=.*[0-9]).{8,}$
665
666        ^              Start anchor
667        (?=.*[A-Z])    Ensure password has one uppercase letters.
668        (?=.*[0-9])    Ensure password has one digit.
669        (?=.*[a-z])    Ensure password has one lowercase letter.
670        .{8,}          Ensure password is of length 8.
671        $              End anchor.
672        """
673
674
675class IUsersContainer(IKofaObject):
676    """A container for officers.
677    """
678
679    def addUser(name, password, title=None, description=None):
680        """Add a user.
681        """
682
683    def delUser(name):
684        """Delete a user if it exists.
685        """
686
687class ILocalRolesAssignable(Interface):
688    """The local roles assignable to an object.
689    """
690    def __call__():
691        """Returns a list of dicts.
692
693        Each dict contains a ``name`` referring to the role assignable
694        for the specified object and a `title` to describe the range
695        of users to which this role can be assigned.
696        """
697
698class IConfigurationContainer(IKofaObject):
699    """A container for session configuration objects.
700    """
701
702    name = schema.TextLine(
703        title = _(u'Name of University'),
704        default = _(u'Sample University'),
705        required = True,
706        )
707
708    acronym = schema.TextLine(
709        title = _(u'Abbreviated Title of University'),
710        default = u'WAeUP.Kofa',
711        required = True,
712        )
713
714    frontpage = schema.Text(
715        title = _(u'Content in HTML format'),
716        required = False,
717        default = default_html_frontpage,
718        constraint=validate_html,
719        )
720
721    frontpage_dict = schema.Dict(
722        title = u'Content as language dictionary with values in html format',
723        required = False,
724        default = {},
725        )
726
727    name_admin = schema.TextLine(
728        title = _(u'Name of Administrator'),
729        default = u'Administrator',
730        required = True,
731        )
732
733    email_admin = schema.ASCIILine(
734        title = _(u'Email Address of Administrator'),
735        default = 'contact@waeup.org',
736        required = True,
737        #constraint=validate_email,
738        )
739
740    email_subject = schema.TextLine(
741        title = _(u'Subject of Email to Administrator'),
742        default = _(u'Kofa Contact'),
743        required = True,
744        )
745
746    smtp_mailer = schema.Choice(
747        title = _(u'SMTP mailer to use when sending mail'),
748        vocabulary = 'Mail Delivery Names',
749        default = 'No email service',
750        required = True,
751        )
752
753    captcha = schema.Choice(
754        title = _(u'Captcha used for public registration pages'),
755        source = CaptchaSource(),
756        default = u'No captcha',
757        required = True,
758        )
759
760    carry_over = schema.Bool(
761        title = _(u'Carry-over Course Registration'),
762        default = False,
763        )
764
765    current_academic_session = schema.Choice(
766        title = _(u'Current Academic Session'),
767        description = _(u'Session for which score editing is allowed'),
768        source = academic_sessions_vocab,
769        default = None,
770        required = False,
771        readonly = False,
772        )
773
774    next_matric_integer = schema.Int(
775        title = _(u'Next Matriculation Number Integer'),
776        description = _(u'Integer used for constructing the next '
777                         'matriculation number'),
778        default = 0,
779        readonly = False,
780        required = False,
781        )
782
783    next_matric_integer_2 = schema.Int(
784        title = _(u'Next Matriculation Number Integer 2'),
785        description = _(u'2nd integer used for constructing the next '
786                         'matriculation number'),
787        default = 0,
788        readonly = False,
789        required = False,
790        )
791
792    next_matric_integer_3 = schema.Int(
793        title = _(u'Next Matriculation Number Integer 3'),
794        description = _(u'3rd integer used for constructing the next '
795                         'matriculation number'),
796        default = 0,
797        readonly = False,
798        required = False,
799        )
800
801    next_matric_integer_4 = schema.Int(
802        title = _(u'Next Matriculation Number Integer 4'),
803        description = _(u'4th integer used for constructing the next '
804                         'matriculation number'),
805        default = 0,
806        readonly = False,
807        required = False,
808        )
809
810    export_disabled_message = schema.Text(
811        title = _(u'Export-disabled message'),
812        description = _(u'Message which will show up if an officer tries '
813                         'to export data. All exporters are automatcally '
814                         'disabled if this field is set.'),
815        required = False,
816        )
817
818    maintmode_enabled_by = schema.TextLine(
819        title = _(u'Maintenance Mode enabled by'),
820        default = None,
821        required = False,
822        )
823
824    def addSessionConfiguration(sessionconfiguration):
825        """Add a session configuration object.
826        """
827
828class ISessionConfiguration(IKofaObject):
829    """A session configuration object.
830    """
831
832    academic_session = schema.Choice(
833        title = _(u'Academic Session'),
834        source = academic_sessions_vocab,
835        default = None,
836        required = True,
837        readonly = True,
838        )
839
840    clearance_enabled = schema.Bool(
841        title = _(u'Clearance enabled'),
842        default = False,
843        )
844
845    payment_disabled = schema.List(
846        title = _(u'Payment disabled'),
847        value_type = schema.Choice(
848            source = DisablePaymentGroupSource(),
849            ),
850        required = False,
851        defaultFactory=list,
852        )
853
854    coursereg_deadline = schema.Datetime(
855        title = _(u'Course Reg. Deadline'),
856        required = False,
857        description = _('Example: ') + u'2011-12-31 23:59:59+01:00',
858        )
859
860    clearance_fee = schema.Float(
861        title = _(u'Acceptance Fee'),
862        default = 0.0,
863        required = False,
864        )
865
866    booking_fee = schema.Float(
867        title = _(u'Bed Booking Fee'),
868        default = 0.0,
869        required = False,
870        )
871
872    maint_fee = schema.Float(
873        title = _(u'Rent (fallback)'),
874        default = 0.0,
875        required = False,
876        )
877
878    late_registration_fee = schema.Float(
879        title = _(u'Late Course Reg. Fee'),
880        default = 0.0,
881        required = False,
882        )
883
884    transcript_fee = schema.Float(
885        title = _(u'Transcript Fee'),
886        default = 0.0,
887        required = False,
888        )
889
890    transfer_fee = schema.Float(
891        title = _(u'Transfer Fee'),
892        default = 0.0,
893        required = False,
894        )
895
896    def getSessionString():
897        """Return the session string from the vocabulary.
898        """
899
900
901class ISessionConfigurationAdd(ISessionConfiguration):
902    """A session configuration object in add mode.
903    """
904
905    academic_session = schema.Choice(
906        title = _(u'Academic Session'),
907        source = academic_sessions_vocab,
908        default = None,
909        required = True,
910        readonly = False,
911        )
912
913ISessionConfigurationAdd['academic_session'].order =  ISessionConfiguration[
914    'academic_session'].order
915
916class IDataCenter(IKofaObject):
917    """A data center.
918
919    A data center manages files (uploads, downloads, etc.).
920
921    Beside providing the bare paths needed to keep files, it also
922    provides some helpers to put results of batch processing into
923    well-defined final locations (with well-defined filenames).
924
925    The main use-case is managing of site-related files, i.e. files
926    for import, export etc.
927
928    DataCenters are _not_ meant as storages for object-specific files
929    like passport photographs and similar.
930
931    It is up to the datacenter implementation how to organize data
932    (paths) inside its storage path.
933    """
934    storage = schema.Bytes(
935        title = u'Path to directory where everything is kept.'
936        )
937
938    deleted_path = schema.Bytes(
939        title = u'Path for deleted student data.'
940        )
941
942    graduated_path = schema.Bytes(
943        title = u'Path for deleted graduated student data.'
944        )
945
946    def getPendingFiles(sort='name'):
947        """Get a list of files stored in `storage` sorted by basename.
948        """
949
950    def getFinishedFiles():
951        """Get a list of files stored in `finished` subfolder of `storage`.
952        """
953
954    def setStoragePath(path, move=False, overwrite=False):
955        """Set the path where to store files.
956
957        If `move` is True, move over files from the current location
958        to the new one.
959
960        If `overwrite` is also True, overwrite any already existing
961        files of same name in target location.
962
963        Triggers a DataCenterStorageMovedEvent.
964        """
965
966    def distProcessedFiles(successful, source_path, finished_file,
967                           pending_file, mode='create', move_orig=True):
968        """Distribute processed files over final locations.
969        """
970
971
972class IDataCenterFile(Interface):
973    """A data center file.
974    """
975
976    name = schema.TextLine(
977        title = u'Filename')
978
979    size = schema.TextLine(
980        title = u'Human readable file size')
981
982    uploaddate = schema.TextLine(
983        title = u'Human readable upload datetime')
984
985    lines = schema.Int(
986        title = u'Number of lines in file')
987
988    def getDate():
989        """Get creation timestamp from file in human readable form.
990        """
991
992    def getSize():
993        """Get human readable size of file.
994        """
995
996    def getLinesNumber():
997        """Get number of lines of file.
998        """
999
1000class IDataCenterStorageMovedEvent(IObjectEvent):
1001    """Emitted, when the storage of a datacenter changes.
1002    """
1003
1004class IObjectUpgradeEvent(IObjectEvent):
1005    """Can be fired, when an object shall be upgraded.
1006    """
1007
1008class ILocalRoleSetEvent(IObjectEvent):
1009    """A local role was granted/revoked for a principal on an object.
1010    """
1011    role_id = Attribute(
1012        "The role id that was set.")
1013    principal_id = Attribute(
1014        "The principal id for which the role was granted/revoked.")
1015    granted = Attribute(
1016        "Boolean. If false, then the role was revoked.")
1017
1018class IQueryResultItem(Interface):
1019    """An item in a search result.
1020    """
1021    url = schema.TextLine(
1022        title = u'URL that links to the found item')
1023    title = schema.TextLine(
1024        title = u'Title displayed in search results.')
1025    description = schema.Text(
1026        title = u'Longer description of the item found.')
1027
1028class IKofaPluggable(Interface):
1029    """A component that might be plugged into a Kofa Kofa app.
1030
1031    Components implementing this interface are referred to as
1032    'plugins'. They are normally called when a new
1033    :class:`waeup.kofa.app.University` instance is created.
1034
1035    Plugins can setup and update parts of the central site without the
1036    site object (normally a :class:`waeup.kofa.app.University` object)
1037    needing to know about that parts. The site simply collects all
1038    available plugins, calls them and the plugins care for their
1039    respective subarea like the applicants area or the datacenter
1040    area.
1041
1042    Currently we have no mechanism to define an order of plugins. A
1043    plugin should therefore make no assumptions about the state of the
1044    site or other plugins being run before and instead do appropriate
1045    checks if necessary.
1046
1047    Updates can be triggered for instance by the respective form in
1048    the site configuration. You normally do updates when the
1049    underlying software changed.
1050    """
1051    def setup(site, name, logger):
1052        """Create an instance of the plugin.
1053
1054        The method is meant to be called by the central app (site)
1055        when it is created.
1056
1057        `site`:
1058           The site that requests a setup.
1059
1060        `name`:
1061           The name under which the plugin was registered (utility name).
1062
1063        `logger`:
1064           A standard Python logger for the plugins use.
1065        """
1066
1067    def update(site, name, logger):
1068        """Method to update an already existing plugin.
1069
1070        This might be called by a site when something serious
1071        changes. It is a poor-man replacement for Zope generations
1072        (but probably more comprehensive and better understandable).
1073
1074        `site`:
1075           The site that requests an update.
1076
1077        `name`:
1078           The name under which the plugin was registered (utility name).
1079
1080        `logger`:
1081           A standard Python logger for the plugins use.
1082        """
1083
1084class IAuthPluginUtility(Interface):
1085    """A component that cares for authentication setup at site creation.
1086
1087    Utilities providing this interface are looked up when a Pluggable
1088    Authentication Utility (PAU) for any
1089    :class:`waeup.kofa.app.University` instance is created and put
1090    into ZODB.
1091
1092    The setup-code then calls the `register` method of the utility and
1093    expects a modified (or unmodified) version of the PAU back.
1094
1095    This allows to define any authentication setup modifications by
1096    submodules or third-party modules/packages.
1097    """
1098
1099    def register(pau):
1100        """Register any plugins wanted to be in the PAU.
1101        """
1102
1103    def unregister(pau):
1104        """Unregister any plugins not wanted to be in the PAU.
1105        """
1106
1107class IObjectConverter(Interface):
1108    """Object converters are available as simple adapters, adapting
1109       interfaces (not regular instances).
1110
1111    """
1112
1113    def fromStringDict(self, data_dict, context, form_fields=None):
1114        """Convert values in `data_dict`.
1115
1116        Converts data in `data_dict` into real values based on
1117        `context` and `form_fields`.
1118
1119        `data_dict` is a mapping (dict) from field names to values
1120        represented as strings.
1121
1122        The fields (keys) to convert can be given in optional
1123        `form_fields`. If given, form_fields should be an instance of
1124        :class:`zope.formlib.form.Fields`. Suitable instances are for
1125        example created by :class:`grok.AutoFields`.
1126
1127        If no `form_fields` are given, a default is computed from the
1128        associated interface.
1129
1130        The `context` can be an existing object (implementing the
1131        associated interface) or a factory name. If it is a string, we
1132        try to create an object using
1133        :func:`zope.component.createObject`.
1134
1135        Returns a tuple ``(<FIELD_ERRORS>, <INVARIANT_ERRORS>,
1136        <DATA_DICT>)`` where
1137
1138        ``<FIELD_ERRORS>``
1139           is a list of tuples ``(<FIELD_NAME>, <ERROR>)`` for each
1140           error that happened when validating the input data in
1141           `data_dict`
1142
1143        ``<INVARIANT_ERRORS>``
1144           is a list of invariant errors concerning several fields
1145
1146        ``<DATA_DICT>``
1147           is a dict with the values from input dict converted.
1148
1149        If errors happen, i.e. the error lists are not empty, always
1150        an empty ``<DATA_DICT>`` is returned.
1151
1152        If ``<DATA_DICT>`` is non-empty, there were no errors.
1153        """
1154
1155class IFieldConverter(Interface):
1156    def request_data(name, value, schema_field, prefix='', mode='create'):
1157        """Create a dict with key-value mapping as created by a request.
1158
1159        `name` and `value` are expected to be parsed from CSV or a
1160        similar input and represent an attribute to be set to a
1161        representation of value.
1162
1163        `mode` gives the mode of import.
1164
1165        :meth:`update_request_data` is then requested to turn this
1166        name and value into vars as they would be sent by a regular
1167        form submit. This means we do not create the real values to be
1168        set but we only define the values that would be sent in a
1169        browser request to request the creation of those values.
1170
1171        The returned dict should contain names and values of a faked
1172        browser request for the given `schema_field`.
1173
1174        Field converters are normally registered as adapters to some
1175        specific zope.schema field.
1176        """
1177
1178class IObjectHistory(Interface):
1179
1180    messages = schema.List(
1181        title = u'List of messages stored',
1182        required = True,
1183        )
1184
1185    def addMessage(message):
1186        """Add a message.
1187        """
1188
1189class IKofaWorkflowInfo(IWorkflowInfo):
1190    """A :class:`hurry.workflow.workflow.WorkflowInfo` with additional
1191       methods for convenience.
1192    """
1193    def getManualTransitions():
1194        """Get allowed manual transitions.
1195
1196        Get a sorted list of tuples containing the `transition_id` and
1197        `title` of each allowed transition.
1198        """
1199
1200class ISiteLoggers(Interface):
1201
1202    loggers = Attribute("A list or generator of registered KofaLoggers")
1203
1204    def register(name, filename=None, site=None, **options):
1205        """Register a logger `name` which logs to `filename`.
1206
1207        If `filename` is not given, logfile will be `name` with
1208        ``.log`` as filename extension.
1209        """
1210
1211    def unregister(name):
1212        """Unregister a once registered logger.
1213        """
1214
1215class ILogger(Interface):
1216    """A logger cares for setup, update and restarting of a Python logger.
1217    """
1218
1219    logger = Attribute("""A :class:`logging.Logger` instance""")
1220
1221
1222    def __init__(name, filename=None, site=None, **options):
1223        """Create a Kofa logger instance.
1224        """
1225
1226    def setup():
1227        """Create a Python :class:`logging.Logger` instance.
1228
1229        The created logger is based on the params given by constructor.
1230        """
1231
1232    def update(**options):
1233        """Update the logger.
1234
1235        Updates the logger respecting modified `options` and changed
1236        paths.
1237        """
1238
1239class ILoggerCollector(Interface):
1240
1241    def getLoggers(site):
1242        """Return all loggers registered for `site`.
1243        """
1244
1245    def registerLogger(site, logging_component):
1246        """Register a logging component residing in `site`.
1247        """
1248
1249    def unregisterLogger(site, logging_component):
1250        """Unregister a logger.
1251        """
1252
1253#
1254# External File Storage and relatives
1255#
1256class IFileStoreNameChooser(INameChooser):
1257    """See zope.container.interfaces.INameChooser for base methods.
1258    """
1259    def checkName(name, attr=None):
1260        """Check whether an object name is valid.
1261
1262        Raises a user error if the name is not valid.
1263        """
1264
1265    def chooseName(name, attr=None):
1266        """Choose a unique valid file id for the object.
1267
1268        The given name may be taken into account when choosing the
1269        name (file id).
1270
1271        chooseName is expected to always choose a valid file id (that
1272        would pass the checkName test) and never raise an error.
1273
1274        If `attr` is not ``None`` it might been taken into account as
1275        well when generating the file id. Usual behaviour is to
1276        interpret `attr` as a hint for what type of file for a given
1277        context should be stored if there are several types
1278        possible. For instance for a certain student some file could
1279        be the connected passport photograph or some certificate scan
1280        or whatever. Each of them has to be stored in a different
1281        location so setting `attr` to a sensible value should give
1282        different file ids returned.
1283        """
1284
1285class IExtFileStore(IFileRetrieval):
1286    """A file storage that stores files in filesystem (not as blobs).
1287    """
1288    root = schema.TextLine(
1289        title = u'Root path of file store.',
1290        )
1291
1292    def getFile(file_id):
1293        """Get raw file data stored under file with `file_id`.
1294
1295        Returns a file descriptor open for reading or ``None`` if the
1296        file cannot be found.
1297        """
1298
1299    def getFileByContext(context, attr=None):
1300        """Get raw file data stored for the given context.
1301
1302        Returns a file descriptor open for reading or ``None`` if no
1303        such file can be found.
1304
1305        Both, `context` and `attr` might be used to find (`context`)
1306        and feed (`attr`) an appropriate file name chooser.
1307
1308        This is a convenience method.
1309        """
1310
1311    def deleteFile(file_id):
1312        """Delete file stored under `file_id`.
1313
1314        Remove file from filestore so, that it is not available
1315        anymore on next call to getFile for the same file_id.
1316
1317        Should not complain if no such file exists.
1318        """
1319
1320    def deleteFileByContext(context, attr=None):
1321        """Delete file for given `context` and `attr`.
1322
1323        Both, `context` and `attr` might be used to find (`context`)
1324        and feed (`attr`) an appropriate file name chooser.
1325
1326        This is a convenience method.
1327        """
1328
1329    def createFile(filename, f):
1330        """Create file given by f with filename `filename`
1331
1332        Returns a hurry.file.File-based object.
1333        """
1334
1335class IFileStoreHandler(Interface):
1336    """Filestore handlers handle specific files for file stores.
1337
1338    If a file to store/get provides a specific filename, a file store
1339    looks up special handlers for that type of file.
1340
1341    """
1342    def pathFromFileID(store, root, filename):
1343        """Turn file id into path to store.
1344
1345        Returned path should be absolute.
1346        """
1347
1348    def createFile(store, root, filename, file_id, file):
1349        """Return some hurry.file based on `store` and `file_id`.
1350
1351        Some kind of callback method called by file stores to create
1352        file objects from file_id.
1353
1354        Returns a tuple ``(raw_file, path, file_like_obj)`` where the
1355        ``file_like_obj`` should be a HurryFile, a KofaImageFile or
1356        similar. ``raw_file`` is the (maybe changed) input file and
1357        ``path`` the relative internal path to store the file at.
1358
1359        Please make sure the ``raw_file`` is opened for reading and
1360        the file descriptor set at position 0 when returned.
1361
1362        This method also gets the raw input file object that is about
1363        to be stored and is expected to raise any exceptions if some
1364        kind of validation or similar fails.
1365        """
1366
1367class IPDF(Interface):
1368    """A PDF representation of some context.
1369    """
1370
1371    def __call__(view=None, note=None):
1372        """Create a bytestream representing a PDF from context.
1373
1374        If `view` is passed in additional infos might be rendered into
1375        the document.
1376
1377        `note` is optional HTML rendered at bottom of the created
1378        PDF. Please consider the limited reportlab support for HTML,
1379        but using font-tags and friends you certainly can get the
1380        desired look.
1381        """
1382
1383class IMailService(Interface):
1384    """A mail service.
1385    """
1386
1387    def __call__():
1388        """Get the default mail delivery.
1389        """
1390
1391
1392class IDataCenterConfig(Interface):
1393    path = Path(
1394        title = u'Path',
1395        description = u"Directory where the datacenter should store "
1396                      u"files by default (adjustable in web UI).",
1397        required = True,
1398        )
1399
1400#
1401# Asynchronous job handling and related
1402#
1403class IJobManager(IKofaObject):
1404    """A manager for asynchronous running jobs (tasks).
1405    """
1406    def put(job, site=None):
1407        """Put a job into task queue.
1408
1409        If no `site` is given, queue job in context of current local
1410        site.
1411
1412        Returns a job_id to identify the put job. This job_id is
1413        needed for further references to the job.
1414        """
1415
1416    def jobs(site=None):
1417        """Get an iterable of jobs stored.
1418        """
1419
1420    def get(job_id, site=None):
1421        """Get the job with id `job_id`.
1422
1423        For the `site` parameter see :meth:`put`.
1424        """
1425
1426    def remove(job_id, site=None):
1427        """Remove job with `job_id` from stored jobs.
1428        """
1429
1430    def start_test_job(site=None):
1431        """Start a test job.
1432        """
1433
1434class IProgressable(Interface):
1435    """A component that can indicate its progress status.
1436    """
1437    percent = schema.Float(
1438        title = u'Percent of job done already.',
1439        )
1440
1441class IJobContainer(IContainer):
1442    """A job container contains IJob objects.
1443    """
1444
1445class IExportJob(zc.async.interfaces.IJob):
1446    def __init__(site, exporter_name):
1447        pass
1448
1449    finished = schema.Bool(
1450        title = u'`True` if the job finished.`',
1451        default = False,
1452        )
1453
1454    failed = schema.Bool(
1455        title = u"`True` iff the job finished and didn't provide a file.",
1456        default = None,
1457        )
1458
1459class IExportJobContainer(IKofaObject):
1460    """A component that contains (maybe virtually) export jobs.
1461    """
1462    def start_export_job(exporter_name, user_id, *args, **kwargs):
1463        """Start asynchronous export job.
1464
1465        `exporter_name` is the name of an exporter utility to be used.
1466
1467        `user_id` is the ID of the user that triggers the export.
1468
1469        `args` positional arguments passed to the export job created.
1470
1471        `kwargs` keyword arguments passed to the export job.
1472
1473        The job_id is stored along with exporter name and user id in a
1474        persistent list.
1475
1476        Returns the job ID of the job started.
1477        """
1478
1479    def get_running_export_jobs(user_id=None):
1480        """Get export jobs for user with `user_id` as list of tuples.
1481
1482        Each tuples holds ``<job_id>, <exporter_name>, <user_id>`` in
1483        that order. The ``<exporter_name>`` is the utility name of the
1484        used exporter.
1485
1486        If `user_id` is ``None``, all running jobs are returned.
1487        """
1488
1489    def get_export_jobs_status(user_id=None):
1490        """Get running/completed export jobs for `user_id` as list of tuples.
1491
1492        Each tuple holds ``<raw status>, <status translated>,
1493        <exporter title>`` in that order, where ``<status
1494        translated>`` and ``<exporter title>`` are translated strings
1495        representing the status of the job and the human readable
1496        title of the exporter used.
1497        """
1498
1499    def delete_export_entry(entry):
1500        """Delete the export denoted by `entry`.
1501
1502        Removes `entry` from the local `running_exports` list and also
1503        removes the regarding job via the local job manager.
1504
1505        `entry` is a tuple ``(<job id>, <exporter name>, <user id>)``
1506        as created by :meth:`start_export_job` or returned by
1507        :meth:`get_running_export_jobs`.
1508        """
1509
1510    def entry_from_job_id(job_id):
1511        """Get entry tuple for `job_id`.
1512
1513        Returns ``None`` if no such entry can be found.
1514        """
1515
1516class IExportContainerFinder(Interface):
1517    """A finder for the central export container.
1518    """
1519    def __call__():
1520        """Return the currently used global or site-wide IExportContainer.
1521        """
1522
1523class IFilteredQuery(IKofaObject):
1524    """A query for objects.
1525    """
1526
1527    defaults = schema.Dict(
1528        title = u'Default Parameters',
1529        required = True,
1530        )
1531
1532    def __init__(**parameters):
1533        """Instantiate a filtered query by passing in parameters.
1534        """
1535
1536    def query():
1537        """Get an iterable of objects denoted by the set parameters.
1538
1539        The search should be applied to objects inside current
1540        site. It's the caller's duty to set the correct site before.
1541
1542        Result can be any iterable like a catalog result set, a list,
1543        or similar.
1544        """
1545
1546class IFilteredCatalogQuery(IFilteredQuery):
1547    """A catalog-based query for objects.
1548    """
1549
1550    cat_name = schema.TextLine(
1551        title = u'Registered name of the catalog to search.',
1552        required = True,
1553        )
1554
1555    def query_catalog(catalog):
1556        """Query catalog with the parameters passed to constructor.
1557        """
Note: See TracBrowser for help on using the repository browser.